@dollhousemcp/mcp-server 2.0.12-rc.9 → 2.0.13

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 (69) hide show
  1. package/CHANGELOG.md +66 -0
  2. package/README.md +4 -4
  3. package/README.md.backup +5 -5
  4. package/README.npm.md +4 -4
  5. package/dist/di/Container.d.ts +14 -0
  6. package/dist/di/Container.d.ts.map +1 -1
  7. package/dist/di/Container.js +43 -26
  8. package/dist/elements/agents/AgentManager.d.ts +11 -4
  9. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  10. package/dist/elements/agents/AgentManager.js +38 -11
  11. package/dist/elements/base/BaseElementManager.d.ts +10 -0
  12. package/dist/elements/base/BaseElementManager.d.ts.map +1 -1
  13. package/dist/elements/base/BaseElementManager.js +26 -1
  14. package/dist/elements/ensembles/EnsembleManager.d.ts +1 -0
  15. package/dist/elements/ensembles/EnsembleManager.d.ts.map +1 -1
  16. package/dist/elements/ensembles/EnsembleManager.js +13 -2
  17. package/dist/elements/templates/Template.d.ts +33 -1
  18. package/dist/elements/templates/Template.d.ts.map +1 -1
  19. package/dist/elements/templates/Template.js +74 -15
  20. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  21. package/dist/elements/templates/TemplateManager.js +8 -1
  22. package/dist/generated/version.d.ts +2 -2
  23. package/dist/generated/version.d.ts.map +1 -1
  24. package/dist/generated/version.js +3 -3
  25. package/dist/handlers/element-crud/createElement.js +2 -2
  26. package/dist/handlers/mcp-aql/SchemaDispatcher.d.ts.map +1 -1
  27. package/dist/handlers/mcp-aql/SchemaDispatcher.js +8 -1
  28. package/dist/handlers/strategies/EnsembleActivationStrategy.d.ts +3 -0
  29. package/dist/handlers/strategies/EnsembleActivationStrategy.d.ts.map +1 -1
  30. package/dist/handlers/strategies/EnsembleActivationStrategy.js +48 -9
  31. package/dist/index.js +19 -13
  32. package/dist/portfolio/DefaultElementProvider.d.ts +8 -0
  33. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  34. package/dist/portfolio/DefaultElementProvider.js +43 -1
  35. package/dist/security/contentValidator.d.ts +15 -0
  36. package/dist/security/contentValidator.d.ts.map +1 -1
  37. package/dist/security/contentValidator.js +40 -2
  38. package/dist/utils/TemplateRenderer.d.ts +9 -0
  39. package/dist/utils/TemplateRenderer.d.ts.map +1 -1
  40. package/dist/utils/TemplateRenderer.js +21 -1
  41. package/dist/web/console/IngestRoutes.d.ts.map +1 -1
  42. package/dist/web/console/IngestRoutes.js +193 -59
  43. package/dist/web/console/SessionNames.d.ts +18 -5
  44. package/dist/web/console/SessionNames.d.ts.map +1 -1
  45. package/dist/web/console/SessionNames.js +63 -8
  46. package/dist/web/console/StaleProcessRecovery.d.ts.map +1 -1
  47. package/dist/web/console/StaleProcessRecovery.js +5 -4
  48. package/dist/web/console/UnifiedConsole.js +3 -3
  49. package/dist/web/console/consoleToken.js +3 -3
  50. package/dist/web/portDiscovery.d.ts +1 -1
  51. package/dist/web/portDiscovery.d.ts.map +1 -1
  52. package/dist/web/portDiscovery.js +2 -2
  53. package/dist/web/public/app.js +65 -11
  54. package/dist/web/public/index.html +2 -0
  55. package/dist/web/public/logs.js +24 -2
  56. package/dist/web/public/metrics.js +22 -4
  57. package/dist/web/public/sessions.js +55 -8
  58. package/dist/web/public/setup.js +11 -2
  59. package/dist/web/public/styles.css +12 -0
  60. package/dist/web/routes/permissionRoutes.js +2 -2
  61. package/dist/web/routes/setupRoutes.d.ts +67 -1
  62. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  63. package/dist/web/routes/setupRoutes.js +298 -6
  64. package/dist/web/routes.d.ts.map +1 -1
  65. package/dist/web/routes.js +4 -2
  66. package/dist/web/server.d.ts.map +1 -1
  67. package/dist/web/server.js +14 -5
  68. package/package.json +5 -3
  69. package/server.json +2 -2
@@ -34,6 +34,41 @@ function safeParseYaml(content) {
34
34
  }
35
35
  }
36
36
 
37
+ globalThis.DollhouseConsoleUI = globalThis.DollhouseConsoleUI || {};
38
+
39
+ /**
40
+ * Show or update a visible error banner within a tab panel.
41
+ *
42
+ * Creates the banner lazily on first use, then reuses it for later updates.
43
+ *
44
+ * @param {string} targetId - DOM id of the tab panel or container that owns the banner
45
+ * @param {string} bannerId - Stable DOM id for the banner element
46
+ * @param {string} message - User-visible message to render inside the banner
47
+ */
48
+ globalThis.DollhouseConsoleUI.showBanner = function(targetId, bannerId, message) {
49
+ const target = document.getElementById(targetId);
50
+ if (!target) return;
51
+ let banner = document.getElementById(bannerId);
52
+ if (!banner) {
53
+ banner = document.createElement('div');
54
+ banner.id = bannerId;
55
+ banner.className = 'tab-error-banner';
56
+ target.prepend(banner);
57
+ }
58
+ banner.textContent = message;
59
+ banner.hidden = false;
60
+ };
61
+
62
+ /**
63
+ * Hide an existing tab-level error banner without removing its DOM node.
64
+ *
65
+ * @param {string} bannerId - Stable DOM id for the banner element
66
+ */
67
+ globalThis.DollhouseConsoleUI.clearBanner = function(bannerId) {
68
+ const banner = document.getElementById(bannerId);
69
+ if (banner) banner.hidden = true;
70
+ };
71
+
37
72
  (() => {
38
73
  const REPO = 'DollhouseMCP/collection';
39
74
  const BRANCH = 'main';
@@ -74,6 +109,7 @@ function safeParseYaml(content) {
74
109
  // ── Bootstrap ──────────────────────────────────────────────────────────────
75
110
 
76
111
  function mergeCollectionData(data) {
112
+ globalThis.DollhouseConsoleUI?.clearBanner?.('collection-error-banner');
77
113
  const CANONICAL_TYPES = new Set(['agents','personas','skills','templates','memories','ensembles']);
78
114
  collectionElements = Object.entries(data.index)
79
115
  .filter(([type]) => CANONICAL_TYPES.has(type))
@@ -117,9 +153,16 @@ function safeParseYaml(content) {
117
153
 
118
154
  // Load community collection (non-blocking — portfolio shows immediately)
119
155
  DollhouseAuth.apiFetch('/api/collection')
120
- .then(r => r.ok ? r.json() : Promise.reject('not available'))
156
+ .then(r => r.ok ? r.json() : Promise.reject(new Error('collection request failed')))
121
157
  .then(mergeCollectionData)
122
- .catch(() => { /* collection not available — portfolio-only mode */ });
158
+ .catch((err) => {
159
+ console.warn('[App] Collection fetch unavailable:', err);
160
+ globalThis.DollhouseConsoleUI?.showBanner?.(
161
+ 'tab-portfolio',
162
+ 'collection-error-banner',
163
+ 'Community collection unavailable — showing local portfolio only.'
164
+ );
165
+ });
123
166
 
124
167
  const updated = document.getElementById('footer-updated');
125
168
  if (updated) {
@@ -1935,11 +1978,18 @@ function safeParseYaml(content) {
1935
1978
 
1936
1979
  const TAB_KEY = 'dollhousemcp-active-tab';
1937
1980
  const SETUP_SEEN_KEY = 'dollhousemcp-setup-seen';
1981
+ // Server version injected at request time — used to show Setup tab once per version
1982
+ // so upgraders automatically see it on each new release (not just first-ever visit).
1983
+ // Validate format (semver-like) before trusting the value; malformed falls back to
1984
+ // 'unknown' which safely triggers setup on every load rather than silently skipping.
1985
+ const _rawVersion = document.querySelector('meta[name="dollhouse-server-version"]')?.content || '';
1986
+ const currentServerVersion = /^\d+\.\d+\.\d+/.test(_rawVersion) ? _rawVersion : 'unknown';
1938
1987
 
1939
1988
  // Determine which tab to show on load:
1940
- // 1. Saved tab from last visit (localStorage)
1941
- // 2. Setup tab on first-ever visit
1942
- // 3. Portfolio (HTML default)
1989
+ // 1. URL hash (deep link)
1990
+ // 2. Saved tab from last visit (localStorage)
1991
+ // 3. Setup tab if not seen on this version yet
1992
+ // 4. Portfolio (HTML default)
1943
1993
  const switchToTab = (tabName) => {
1944
1994
  if (!consoleTabs) return;
1945
1995
  const btn = consoleTabs.querySelector(`[data-tab="${tabName}"]`);
@@ -2066,12 +2116,16 @@ function safeParseYaml(content) {
2066
2116
  }
2067
2117
 
2068
2118
  if (!applyHashTab()) {
2069
- const savedTab = localStorage.getItem(TAB_KEY);
2070
- if (savedTab) {
2071
- switchToTab(savedTab);
2072
- lazyInitTab(savedTab, tabInits);
2073
- } else if (!localStorage.getItem(SETUP_SEEN_KEY)) {
2074
- localStorage.setItem(SETUP_SEEN_KEY, '1');
2119
+ // Version check takes priority over saved tab — upgraders must see Setup
2120
+ // regardless of whether they have a saved tab from their previous session.
2121
+ if (localStorage.getItem(SETUP_SEEN_KEY) === currentServerVersion) {
2122
+ const savedTab = localStorage.getItem(TAB_KEY);
2123
+ if (savedTab) {
2124
+ switchToTab(savedTab);
2125
+ lazyInitTab(savedTab, tabInits);
2126
+ }
2127
+ } else {
2128
+ localStorage.setItem(SETUP_SEEN_KEY, currentServerVersion);
2075
2129
  switchToTab('setup');
2076
2130
  }
2077
2131
  }
@@ -12,6 +12,8 @@
12
12
  as an Authorization: Bearer header on fetch calls, or as a ?token= query
13
13
  param on EventSource connections. An empty value means auth is off. -->
14
14
  <meta name="dollhouse-console-token" content="{{CONSOLE_TOKEN}}">
15
+ <!-- Server version — injected at request time for version-aware UI behaviour (e.g. setup-seen per version). -->
16
+ <meta name="dollhouse-server-version" content="{{DOLLHOUSE_VERSION}}">
15
17
  <link rel="stylesheet" href="fonts.css">
16
18
  <link rel="stylesheet" href="styles.css">
17
19
  <link rel="stylesheet" href="logs.css">
@@ -38,6 +38,7 @@
38
38
  let filterSource = '';
39
39
  let filterMessage = '';
40
40
  let filterCorrelationId = '';
41
+ let filterSessionId = '';
41
42
 
42
43
  // ── DOM references ─────────────────────────────────────────────────────
43
44
  let viewport, scrollSpacer, jumpBtn, statusDot, statusText, entryCountEl;
@@ -56,6 +57,14 @@
56
57
  if (autoScroll) scrollToBottom();
57
58
  });
58
59
  },
60
+ refilter: (sessionId) => {
61
+ filterSessionId = sessionId || '';
62
+ applyFilters();
63
+ requestAnimationFrame(() => {
64
+ renderViewport();
65
+ if (autoScroll) scrollToBottom();
66
+ });
67
+ },
59
68
  };
60
69
 
61
70
  function initLogViewer(urlParams) {
@@ -307,7 +316,10 @@
307
316
  const qs = params.toString();
308
317
  eventSource = DollhouseAuth.apiEventSource('/api/logs/stream' + (qs ? '?' + qs : ''));
309
318
 
310
- eventSource.onopen = () => setStatus('connected');
319
+ eventSource.onopen = () => {
320
+ clearLogsError();
321
+ setStatus('connected');
322
+ };
311
323
 
312
324
  eventSource.onmessage = (event) => {
313
325
  try {
@@ -322,6 +334,7 @@
322
334
 
323
335
  eventSource.onerror = () => {
324
336
  setStatus('disconnected');
337
+ showLogsError('Connection lost - reconnecting...');
325
338
  eventSource.close();
326
339
  eventSource = null;
327
340
  setTimeout(connectSSE, RECONNECT_DELAY_MS);
@@ -362,6 +375,7 @@
362
375
  const LEVEL_PRIORITY = { debug: 0, info: 1, warn: 2, error: 3 };
363
376
 
364
377
  function matchesFilters(entry) {
378
+ if (filterSessionId && entry.data?._sessionId !== filterSessionId) return false;
365
379
  if (filterCorrelationId && entry.correlationId !== filterCorrelationId) return false;
366
380
  if (filterCategory && entry.category !== filterCategory) return false;
367
381
  if (filterLevel && (LEVEL_PRIORITY[entry.level] || 0) < (LEVEL_PRIORITY[filterLevel] || 0)) return false;
@@ -371,7 +385,7 @@
371
385
  }
372
386
 
373
387
  function applyFilters() {
374
- const hasFilter = filterCategory || filterLevel || filterSource || filterMessage || filterCorrelationId;
388
+ const hasFilter = filterCategory || filterLevel || filterSource || filterMessage || filterCorrelationId || filterSessionId;
375
389
  if (hasFilter) {
376
390
  filteredIndices = [];
377
391
  for (let i = 0; i < buffer.length; i++) {
@@ -636,6 +650,14 @@
636
650
  statusText.textContent = status;
637
651
  }
638
652
 
653
+ function showLogsError(message) {
654
+ globalThis.DollhouseConsoleUI?.showBanner?.('tab-logs', 'logs-error-banner', message);
655
+ }
656
+
657
+ function clearLogsError() {
658
+ globalThis.DollhouseConsoleUI?.clearBanner?.('logs-error-banner');
659
+ }
660
+
639
661
  function updateEntryCount() {
640
662
  const total = buffer.length;
641
663
  const visible = getVisibleCount();
@@ -144,6 +144,15 @@
144
144
  });
145
145
  }
146
146
 
147
+ // ── Error banners (#1866) ────────────────────────────────────────────────
148
+ function showMetricsError(message) {
149
+ globalThis.DollhouseConsoleUI?.showBanner?.('tab-metrics', 'metrics-error-banner', message);
150
+ }
151
+
152
+ function clearMetricsError() {
153
+ globalThis.DollhouseConsoleUI?.clearBanner?.('metrics-error-banner');
154
+ }
155
+
147
156
  // ── Data fetching ────────────────────────────────────────────────────────
148
157
  async function fetchLatest() {
149
158
  try {
@@ -160,8 +169,12 @@
160
169
  const cutoff = Date.now() - TIME_RANGES['1h'];
161
170
  historySnapshots = historySnapshots.filter(s => new Date(s.timestamp).getTime() > cutoff);
162
171
  renderAll(lastSnapshot.metrics);
172
+ clearMetricsError();
163
173
  }
164
- } catch { /* network error, will retry */ }
174
+ } catch (err) {
175
+ console.warn('[Metrics] Fetch failed:', err);
176
+ showMetricsError('Failed to load metrics — retrying...');
177
+ }
165
178
  }
166
179
 
167
180
  async function fetchHistory() {
@@ -174,7 +187,11 @@
174
187
  historySnapshots = data.snapshots.reverse(); // oldest first
175
188
  if (lastSnapshot) renderAll(lastSnapshot.metrics);
176
189
  }
177
- } catch { /* network error */ }
190
+ clearMetricsError();
191
+ } catch (err) {
192
+ console.warn('[Metrics] History fetch failed:', err);
193
+ showMetricsError('Failed to load metrics history — retrying...');
194
+ }
178
195
  }
179
196
 
180
197
  // ── Rendering ────────────────────────────────────────────────────────────
@@ -412,9 +429,10 @@
412
429
  renderSecurityEvents(data.entries);
413
430
  }
414
431
  })
415
- .catch(() => {
432
+ .catch((err) => {
433
+ console.warn('[Metrics] Security events fetch failed:', err);
416
434
  const el = document.getElementById('security-recent-events');
417
- if (el) el.innerHTML = '';
435
+ if (el) el.textContent = 'Failed to load security events';
418
436
  });
419
437
  }
420
438
  }
@@ -14,7 +14,15 @@
14
14
  (function() {
15
15
  'use strict';
16
16
 
17
- var SESSION_POLL_INTERVAL = 5000;
17
+ function getConfiguredNumber(key, fallback) {
18
+ var config = globalThis.DollhouseConsoleConfig;
19
+ var value = config && Number(config[key]);
20
+ return Number.isFinite(value) && value > 0 ? value : fallback;
21
+ }
22
+
23
+ var SESSION_POLL_INTERVAL = getConfiguredNumber('sessionPollIntervalMs', 5000);
24
+ var SESSION_FILTER_INJECTION_RETRY_INTERVAL = getConfiguredNumber('sessionFilterInjectionRetryIntervalMs', 500);
25
+ var SESSION_FILTER_INJECTION_MAX_RETRIES = getConfiguredNumber('sessionFilterInjectionMaxRetries', 20);
18
26
  var sessions = [];
19
27
  var filterSessionId = '';
20
28
  var dropdownBuilt = false;
@@ -59,14 +67,33 @@
59
67
  var logSelect = document.getElementById('log-session-filter');
60
68
  if (logSelect) logSelect.value = sessionId;
61
69
 
62
- // Trigger log re-filter
70
+ // Trigger log re-filter with the selected session
63
71
  if (window.DollhouseConsole && window.DollhouseConsole.logs && window.DollhouseConsole.logs.refilter) {
64
- window.DollhouseConsole.logs.refilter();
72
+ window.DollhouseConsole.logs.refilter(sessionId);
65
73
  }
66
74
 
67
75
  refreshSelectionState();
68
76
  }
69
77
 
78
+ function showSessionsError(message) {
79
+ var target = document.getElementById('session-indicator');
80
+ if (!target || !target.parentElement) return;
81
+ var banner = document.getElementById('sessions-error-banner');
82
+ if (!banner) {
83
+ banner = document.createElement('div');
84
+ banner.id = 'sessions-error-banner';
85
+ banner.className = 'tab-error-banner';
86
+ target.parentElement.insertBefore(banner, target);
87
+ }
88
+ banner.textContent = message;
89
+ banner.hidden = false;
90
+ }
91
+
92
+ function clearSessionsError() {
93
+ var banner = document.getElementById('sessions-error-banner');
94
+ if (banner) banner.hidden = true;
95
+ }
96
+
70
97
  // Update checkmarks and selected styling without rebuilding DOM
71
98
  function refreshSelectionState() {
72
99
  // Update items
@@ -274,6 +301,15 @@
274
301
  .then(function(res) {
275
302
  if (!res.ok) {
276
303
  alert('Failed to stop session ' + displayName(s) + ': server returned ' + res.status);
304
+ fetchSessions();
305
+ return;
306
+ }
307
+ return res.json();
308
+ })
309
+ .then(function(data) {
310
+ if (!data) return;
311
+ if (data.reason === 'pending-kill') {
312
+ alert('Session ' + displayName(s) + ' will be terminated shortly.\nWaiting for the process to identify itself, then it will be killed.');
277
313
  }
278
314
  fetchSessions();
279
315
  })
@@ -329,7 +365,7 @@
329
365
  if (!logPanel) return;
330
366
  if (document.getElementById('log-session-filter')) return;
331
367
 
332
- var filterBar = logPanel.querySelector('.log-filters');
368
+ var filterBar = logPanel.querySelector('.log-controls');
333
369
  if (!filterBar) return;
334
370
 
335
371
  var group = document.createElement('div');
@@ -343,6 +379,10 @@
343
379
  group.querySelector('select').addEventListener('change', function() {
344
380
  applyFilter(this.value);
345
381
  });
382
+
383
+ // If sessions loaded before the log controls mounted, populate the
384
+ // newly injected filter immediately instead of waiting for the next poll.
385
+ updateSessionFilterOptions();
346
386
  }
347
387
 
348
388
  // Update session filter dropdown options
@@ -369,15 +409,22 @@
369
409
  */
370
410
  function fetchSessions() {
371
411
  DollhouseAuth.apiFetch('/api/sessions').then(function(res) {
372
- if (!res.ok) return;
412
+ if (!res.ok) {
413
+ showSessionsError('Failed to load sessions.');
414
+ return;
415
+ }
373
416
  return res.json();
374
417
  }).then(function(data) {
375
418
  if (data && data.sessions) {
376
419
  sessions = data.sessions;
377
420
  updateSessionIndicator();
378
421
  updateSessionFilterOptions();
422
+ clearSessionsError();
379
423
  }
380
- }).catch(function() {});
424
+ }).catch(function(err) {
425
+ console.warn('[Sessions] Fetch failed:', err);
426
+ showSessionsError('Failed to load sessions.');
427
+ });
381
428
  }
382
429
 
383
430
  // Expose for logs.js integration
@@ -395,10 +442,10 @@
395
442
  var tryInject = setInterval(function() {
396
443
  injectSessionFilter();
397
444
  retries++;
398
- if (document.getElementById('log-session-filter') || retries > 20) {
445
+ if (document.getElementById('log-session-filter') || retries > SESSION_FILTER_INJECTION_MAX_RETRIES) {
399
446
  clearInterval(tryInject);
400
447
  }
401
- }, 500);
448
+ }, SESSION_FILTER_INJECTION_RETRY_INTERVAL);
402
449
  }
403
450
 
404
451
  if (document.readyState === 'loading') {
@@ -176,8 +176,16 @@
176
176
  if (hint) hint.textContent = CHANNEL_HINTS[currentChannel] || '';
177
177
  configs = buildConfigs(pinnedVersion, currentChannel);
178
178
  updateAllConfigs(currentMethod);
179
+ // Clear is-success/is-match state so buttons can be re-evaluated
180
+ document.querySelectorAll('.setup-install-btn').forEach((btn) => {
181
+ btn.classList.remove('is-success', 'is-match');
182
+ btn.disabled = false;
183
+ });
184
+ document.querySelectorAll('.setup-install-status').forEach((s) => {
185
+ s.textContent = '';
186
+ s.className = 'setup-install-status';
187
+ });
179
188
  updateInstallButtonLabels();
180
- // Re-evaluate detection: current config may no longer match the new channel
181
189
  updateDetectionState();
182
190
  });
183
191
  };
@@ -756,7 +764,8 @@
756
764
  installBtn.classList.add('is-match');
757
765
  } else {
758
766
  const isPinned = currentMethod === 'global' && pinnedVersion && pinnedVersion !== 'latest';
759
- installBtn.textContent = isPinned ? `Configure v${pinnedVersion}` : 'Configure Now';
767
+ const channelLabel = currentChannel === DEFAULT_CHANNEL ? '' : ` (${currentChannel})`;
768
+ installBtn.textContent = isPinned ? `Configure v${pinnedVersion}` : `Configure Now${channelLabel}`;
760
769
  installBtn.disabled = false;
761
770
  installBtn.classList.remove('is-match');
762
771
  }
@@ -1727,3 +1727,15 @@ fieldset.topic-filters {
1727
1727
  scroll-behavior: auto !important;
1728
1728
  }
1729
1729
  }
1730
+
1731
+ /* ── Tab error banners (#1866) ─────────────────────────────────────────── */
1732
+ .tab-error-banner {
1733
+ padding: 6px 10px;
1734
+ border-radius: var(--radius-sm);
1735
+ font-size: 11.5px;
1736
+ margin-bottom: 8px;
1737
+ background: rgba(245, 158, 11, 0.1);
1738
+ color: #b45309; /* NOSONAR — contrast computed against page bg, not rgba tint */
1739
+ border: 1px solid rgba(245, 158, 11, 0.2);
1740
+ font-family: var(--font-mono);
1741
+ }
@@ -83,7 +83,7 @@ export function registerPermissionRoutes(router, handler) {
83
83
  res.json({ decision: 'allow' }); // fail open
84
84
  return;
85
85
  }
86
- const decision = opResult.data?.decision ?? 'unknown';
86
+ const decision = (opResult.data?.decision ?? 'unknown');
87
87
  logger.debug(`[WebUI/Gateway] evaluate_permission: ${tool_name} → ${decision} (${elapsedMs}ms)`);
88
88
  // Track decision for live dashboard feed
89
89
  trackDecision(tool_name, input || {}, opResult.data);
@@ -135,4 +135,4 @@ export function registerPermissionRoutes(router, handler) {
135
135
  }
136
136
  });
137
137
  }
138
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionRoutes.js","sourceRoot":"","sources":["../../../src/web/routes/permissionRoutes.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation HTTP routes and decision tracking.\n *\n * Provides:\n * - POST /evaluate_permission — evaluates tool permissions via MCP-AQL\n * - GET /permissions/status — returns current policies and recent decisions\n * - Decision tracking ring buffer for the live dashboard feed\n */\n\nimport express, { Router } from 'express';\nimport { logger } from '../../utils/logger.js';\nimport type { MCPAQLHandler } from '../../handlers/mcp-aql/MCPAQLHandler.js';\n\nimport { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';\n\n// ── Permission Decision Tracking ─────────────────────────────────────────────\n// Ring buffer of recent permission decisions for the live dashboard feed.\n\ninterface PermissionDecision {\n  id: string;\n  timestamp: string;\n  tool_name: string;\n  command?: string;\n  decision: string;\n  reason?: string;\n}\n\nconst DECISION_BUFFER_SIZE = 200;\nconst recentDecisions: PermissionDecision[] = [];\nlet decisionCounter = 0;\n\n/** Extract a string field from a record, trying multiple keys in order */\nfunction extractString(obj: Record<string, unknown>, keys: string[], fallback: string): string {\n  for (const key of keys) {\n    const val = obj?.[key];\n    if (typeof val === 'string') return val;\n  }\n  return fallback;\n}\n\nfunction trackDecision(toolName: string, input: Record<string, unknown>, result: Record<string, unknown>): void {\n  const entry: PermissionDecision = {\n    id: `d-${++decisionCounter}`,\n    timestamp: new Date().toISOString(),\n    tool_name: toolName,\n    command: toolName === 'Bash' && typeof input?.command === 'string' ? input.command : undefined,\n    decision: extractString(result, ['decision', 'behavior'], 'unknown'),\n    reason: extractString(result, ['reason', 'message'], ''),\n  };\n  recentDecisions.unshift(entry);\n  if (recentDecisions.length > DECISION_BUFFER_SIZE) {\n    recentDecisions.length = DECISION_BUFFER_SIZE;\n  }\n}\n\n/** Helper to extract single result from MCP-AQL batch response */\nfunction asSingleResult(results: unknown): { success: boolean; data?: unknown; error?: string } {\n  if (Array.isArray(results)) return results[0] || { success: false, error: 'Empty result' };\n  return results as { success: boolean; data?: unknown; error?: string };\n}\n\n/**\n * Register permission-related routes on a gateway router.\n * Must be called with the MCP-AQL handler for policy evaluation.\n */\nexport function registerPermissionRoutes(router: Router, handler: MCPAQLHandler): void {\n  /**\n   * POST /api/evaluate_permission\n   * Permission evaluation endpoint for PreToolUse hooks.\n   * Routes through evaluate_permission MCP-AQL READ operation.\n   * Fail-open: returns allow on any error to avoid blocking the user.\n   */\n  const permissionLimiter = new SlidingWindowRateLimiter(120, 60_000);\n  router.post('/evaluate_permission', express.json(), async (req, res) => {\n    if (!permissionLimiter.tryAcquire()) {\n      res.json({ decision: 'allow' }); // fail open on rate limit\n      return;\n    }\n\n    const body = req.body as {\n      tool_name?: string;\n      input?: Record<string, unknown>;\n      platform?: string;\n    };\n\n    // Unicode normalization (NFC) on string inputs to prevent homograph attacks\n    const tool_name = typeof body.tool_name === 'string' ? body.tool_name.normalize('NFC') : undefined;\n    const platform = typeof body.platform === 'string' ? body.platform.normalize('NFC') : undefined;\n    const input = body.input;\n\n    if (!tool_name) {\n      res.json({ decision: 'allow' }); // fail open on bad input\n      return;\n    }\n\n    const startMs = Date.now();\n    try {\n      const opResult = asSingleResult(await handler.handleRead({\n        operation: 'evaluate_permission',\n        params: {\n          tool_name,\n          input: input || {},\n          platform: platform || 'claude_code',\n        },\n      }));\n      const elapsedMs = Date.now() - startMs;\n\n      if (!opResult.success) {\n        logger.warn(`[WebUI/Gateway] evaluate_permission failed (${elapsedMs}ms): ${opResult.error}`);\n        res.json({ decision: 'allow' }); // fail open\n        return;\n      }\n\n      const decision = (opResult.data as Record<string, unknown>)?.decision ?? 'unknown';\n      logger.debug(`[WebUI/Gateway] evaluate_permission: ${tool_name} → ${decision} (${elapsedMs}ms)`);\n\n      // Track decision for live dashboard feed\n      trackDecision(tool_name, input || {}, opResult.data as Record<string, unknown>);\n\n      res.json(opResult.data);\n    } catch (err) {\n      const elapsedMs = Date.now() - startMs;\n      logger.error(`[WebUI/Gateway] evaluate_permission error (${elapsedMs}ms):`, err);\n      res.json({ decision: 'allow' }); // fail open\n    }\n  });\n\n  /**\n   * GET /api/permissions/status\n   * Returns current permission policies and recent decisions\n   * for the live permissions dashboard.\n   */\n  router.get('/permissions/status', async (_req, res) => {\n    try {\n      const opResult = asSingleResult(await handler.handleRead({\n        operation: 'get_effective_cli_policies',\n      }));\n\n      if (!opResult.success) {\n        res.status(500).json({ error: opResult.error || 'Failed to get policies' });\n        return;\n      }\n\n      const data = opResult.data as Record<string, unknown>;\n\n      // Extract confirm patterns from elements\n      const elements = (data.elements || []) as Array<Record<string, unknown>>;\n      const confirmPatterns: string[] = [];\n      for (const el of elements) {\n        const confirm = el.confirmPatterns as string[] | undefined;\n        if (confirm?.length) confirmPatterns.push(...confirm);\n      }\n\n      res.json({\n        activeElementCount: data.activeElementCount,\n        hasAllowlist: data.hasAllowlist,\n        denyPatterns: data.combinedDenyPatterns,\n        allowPatterns: data.combinedAllowPatterns,\n        confirmPatterns,\n        elements,\n        permissionPromptActive: data.permissionPromptActive,\n        recentDecisions,\n      });\n    } catch (err) {\n      logger.error('[WebUI/Gateway] permissions/status error:', err);\n      res.status(500).json({ error: 'Failed to get permission status' });\n    }\n  });\n}\n"]}
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionRoutes.js","sourceRoot":"","sources":["../../../src/web/routes/permissionRoutes.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Permission evaluation HTTP routes and decision tracking.\n *\n * Provides:\n * - POST /evaluate_permission — evaluates tool permissions via MCP-AQL\n * - GET /permissions/status — returns current policies and recent decisions\n * - Decision tracking ring buffer for the live dashboard feed\n */\n\nimport express, { Router } from 'express';\nimport { logger } from '../../utils/logger.js';\nimport type { MCPAQLHandler } from '../../handlers/mcp-aql/MCPAQLHandler.js';\n\nimport { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';\n\n// ── Permission Decision Tracking ─────────────────────────────────────────────\n// Ring buffer of recent permission decisions for the live dashboard feed.\n\ninterface PermissionDecision {\n  id: string;\n  timestamp: string;\n  tool_name: string;\n  command?: string;\n  decision: string;\n  reason?: string;\n}\n\nconst DECISION_BUFFER_SIZE = 200;\nconst recentDecisions: PermissionDecision[] = [];\nlet decisionCounter = 0;\n\n/** Extract a string field from a record, trying multiple keys in order */\nfunction extractString(obj: Record<string, unknown>, keys: string[], fallback: string): string {\n  for (const key of keys) {\n    const val = obj?.[key];\n    if (typeof val === 'string') return val;\n  }\n  return fallback;\n}\n\nfunction trackDecision(toolName: string, input: Record<string, unknown>, result: Record<string, unknown>): void {\n  const entry: PermissionDecision = {\n    id: `d-${++decisionCounter}`,\n    timestamp: new Date().toISOString(),\n    tool_name: toolName,\n    command: toolName === 'Bash' && typeof input?.command === 'string' ? input.command : undefined,\n    decision: extractString(result, ['decision', 'behavior'], 'unknown'),\n    reason: extractString(result, ['reason', 'message'], ''),\n  };\n  recentDecisions.unshift(entry);\n  if (recentDecisions.length > DECISION_BUFFER_SIZE) {\n    recentDecisions.length = DECISION_BUFFER_SIZE;\n  }\n}\n\n/** Helper to extract single result from MCP-AQL batch response */\nfunction asSingleResult(results: unknown): { success: boolean; data?: unknown; error?: string } {\n  if (Array.isArray(results)) return results[0] || { success: false, error: 'Empty result' };\n  return results as { success: boolean; data?: unknown; error?: string };\n}\n\n/**\n * Register permission-related routes on a gateway router.\n * Must be called with the MCP-AQL handler for policy evaluation.\n */\nexport function registerPermissionRoutes(router: Router, handler: MCPAQLHandler): void {\n  /**\n   * POST /api/evaluate_permission\n   * Permission evaluation endpoint for PreToolUse hooks.\n   * Routes through evaluate_permission MCP-AQL READ operation.\n   * Fail-open: returns allow on any error to avoid blocking the user.\n   */\n  const permissionLimiter = new SlidingWindowRateLimiter(120, 60_000);\n  router.post('/evaluate_permission', express.json(), async (req, res) => {\n    if (!permissionLimiter.tryAcquire()) {\n      res.json({ decision: 'allow' }); // fail open on rate limit\n      return;\n    }\n\n    const body = req.body as {\n      tool_name?: string;\n      input?: Record<string, unknown>;\n      platform?: string;\n    };\n\n    // Unicode normalization (NFC) on string inputs to prevent homograph attacks\n    const tool_name = typeof body.tool_name === 'string' ? body.tool_name.normalize('NFC') : undefined;\n    const platform = typeof body.platform === 'string' ? body.platform.normalize('NFC') : undefined;\n    const input = body.input;\n\n    if (!tool_name) {\n      res.json({ decision: 'allow' }); // fail open on bad input\n      return;\n    }\n\n    const startMs = Date.now();\n    try {\n      const opResult = asSingleResult(await handler.handleRead({\n        operation: 'evaluate_permission',\n        params: {\n          tool_name,\n          input: input || {},\n          platform: platform || 'claude_code',\n        },\n      }));\n      const elapsedMs = Date.now() - startMs;\n\n      if (!opResult.success) {\n        logger.warn(`[WebUI/Gateway] evaluate_permission failed (${elapsedMs}ms): ${opResult.error}`);\n        res.json({ decision: 'allow' }); // fail open\n        return;\n      }\n\n      const decision = ((opResult.data as { decision?: string })?.decision ?? 'unknown');\n      logger.debug(`[WebUI/Gateway] evaluate_permission: ${tool_name} → ${decision} (${elapsedMs}ms)`);\n\n      // Track decision for live dashboard feed\n      trackDecision(tool_name, input || {}, opResult.data as Record<string, unknown>);\n\n      res.json(opResult.data);\n    } catch (err) {\n      const elapsedMs = Date.now() - startMs;\n      logger.error(`[WebUI/Gateway] evaluate_permission error (${elapsedMs}ms):`, err);\n      res.json({ decision: 'allow' }); // fail open\n    }\n  });\n\n  /**\n   * GET /api/permissions/status\n   * Returns current permission policies and recent decisions\n   * for the live permissions dashboard.\n   */\n  router.get('/permissions/status', async (_req, res) => {\n    try {\n      const opResult = asSingleResult(await handler.handleRead({\n        operation: 'get_effective_cli_policies',\n      }));\n\n      if (!opResult.success) {\n        res.status(500).json({ error: opResult.error || 'Failed to get policies' });\n        return;\n      }\n\n      const data = opResult.data as Record<string, unknown>;\n\n      // Extract confirm patterns from elements\n      const elements = (data.elements || []) as Array<Record<string, unknown>>;\n      const confirmPatterns: string[] = [];\n      for (const el of elements) {\n        const confirm = el.confirmPatterns as string[] | undefined;\n        if (confirm?.length) confirmPatterns.push(...confirm);\n      }\n\n      res.json({\n        activeElementCount: data.activeElementCount,\n        hasAllowlist: data.hasAllowlist,\n        denyPatterns: data.combinedDenyPatterns,\n        allowPatterns: data.combinedAllowPatterns,\n        confirmPatterns,\n        elements,\n        permissionPromptActive: data.permissionPromptActive,\n        recentDecisions,\n      });\n    } catch (err) {\n      logger.error('[WebUI/Gateway] permissions/status error:', err);\n      res.status(500).json({ error: 'Failed to get permission status' });\n    }\n  });\n}\n"]}
@@ -8,7 +8,12 @@
8
8
  * and command arguments are hardcoded — no user-supplied shell input.
9
9
  */
10
10
  import type { Request, Response } from 'express';
11
- export declare function createSetupRoutes(): {
11
+ export declare function createSetupRoutes(opts?: {
12
+ /** Override install-mcp runner. For testing only — prefix signals test-only use. */
13
+ _runInstallMcp?: (client: string, version?: string) => Promise<string>;
14
+ /** Skip the sliding-window rate limiter. For testing only. */
15
+ _skipRateLimit?: boolean;
16
+ }): {
12
17
  installHandler: (req: Request, res: Response) => Promise<void>;
13
18
  openConfigHandler: (req: Request, res: Response) => Promise<void>;
14
19
  versionHandler: (req: Request, res: Response) => Promise<void>;
@@ -19,4 +24,65 @@ export declare function createSetupRoutes(): {
19
24
  verifyLicenseHandler: (req: Request, res: Response) => Promise<void>;
20
25
  resendVerificationHandler: (req: Request, res: Response) => Promise<void>;
21
26
  };
27
+ /** Result of attempting to apply the NVM launcher mitigation. */
28
+ export type NvmLauncherResult = 'applied' | 'not-applicable' | 'failed';
29
+ /**
30
+ * Orchestrates the NVM mitigation: detect → create launcher → patch config → telemetry.
31
+ * Extracted from installHandler to keep its cognitive complexity within SonarCloud limits.
32
+ * Returns a result enum rather than throwing so the caller always gets a clean signal.
33
+ *
34
+ * @param home - Override home directory (injectable for tests)
35
+ */
36
+ export declare function applyNvmLauncherIfNeeded(client: string, home?: string): Promise<NvmLauncherResult>;
37
+ /**
38
+ * Startup repair: re-creates the wrapper and re-patches all known JSON-format
39
+ * client configs on every server start. Handles two cases:
40
+ * 1. Wrapper was deleted — recreates it so configs pointing to it keep working.
41
+ * 2. Pre-existing install (user installed before this fix shipped) — patches
42
+ * configs that still use bare `npx`.
43
+ *
44
+ * Fire-and-forget from startWebServer. All errors are swallowed and logged.
45
+ *
46
+ * @param home - Override home directory (injectable for tests)
47
+ * @param configPathResolver - Override config path lookup (injectable for tests).
48
+ * Return null to skip a client entirely.
49
+ * Defaults to the production getConfigPath.
50
+ */
51
+ export declare function repairNvmLauncherOnStartup(home?: string, configPathResolver?: (client: string) => string | null): Promise<void>;
52
+ /**
53
+ * Returns true if NVM is installed on this machine (macOS/Linux only).
54
+ * Checks process.env.NVM_DIR first (handles non-standard install locations),
55
+ * then falls back to ~/.nvm.
56
+ *
57
+ * @param home - Override home directory (defaults to os.homedir(); injectable for tests)
58
+ */
59
+ export declare function isNvmPresent(home?: string): Promise<boolean>;
60
+ /**
61
+ * Creates ~/.dollhouse/bin/dollhousemcp-nvm.sh and returns its path.
62
+ *
63
+ * The NVM directory is resolved at generation time and hardcoded into the
64
+ * script. This is intentional: Claude Desktop does not source the user's
65
+ * shell profile, so $NVM_DIR would be unset when the wrapper runs. By
66
+ * embedding the absolute path we guarantee the correct NVM is found.
67
+ *
68
+ * The script sources NVM, then checks the active Node major version. If it
69
+ * is below 18 (the DollhouseMCP minimum), it tries `nvm use node` (highest
70
+ * installed) then `nvm use --lts` as a fallback. A final version check
71
+ * writes a warning to stderr if the node is still too old — that warning
72
+ * will appear in Claude Desktop's error log.
73
+ *
74
+ * @param home - Override home directory (injectable for tests)
75
+ * @param nvmDirOverride - Override the resolved NVM path (injectable for tests)
76
+ */
77
+ export declare function ensureNvmLauncher(home?: string, nvmDirOverride?: string): Promise<string>;
78
+ /**
79
+ * Patches the dollhousemcp entry in an MCP client's JSON config to use
80
+ * the NVM-aware launcher instead of bare `npx`.
81
+ *
82
+ * Only acts on JSON-format configs. TOML configs (codex) are skipped.
83
+ * Silently no-ops if the config file is missing or unreadable.
84
+ *
85
+ * @param configPathOverride - Use this path instead of the platform default (injectable for tests)
86
+ */
87
+ export declare function patchConfigForNvmLauncher(client: string, wrapperPath: string, configPathOverride?: string): Promise<void>;
22
88
  //# sourceMappingURL=setupRoutes.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"setupRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/setupRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAkWjD,wBAAgB,iBAAiB,IAAI;IACnC,cAAc,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/D,iBAAiB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClE,cAAc,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/D,mBAAmB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IACpE,aAAa,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC9D,iBAAiB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClE,iBAAiB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClE,oBAAoB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IACrE,yBAAyB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;CAC3E,CAqbA"}
1
+ {"version":3,"file":"setupRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/routes/setupRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,SAAS,CAAC;AAkWjD,wBAAgB,iBAAiB,CAAC,IAAI,CAAC,EAAE;IACvC,oFAAoF;IACpF,cAAc,CAAC,EAAE,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,MAAM,KAAK,OAAO,CAAC,MAAM,CAAC,CAAC;IACvE,8DAA8D;IAC9D,cAAc,CAAC,EAAE,OAAO,CAAC;CAC1B,GAAG;IACF,cAAc,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/D,iBAAiB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClE,cAAc,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/D,mBAAmB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IACpE,aAAa,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAC9D,iBAAiB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClE,iBAAiB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAClE,oBAAoB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IACrE,yBAAyB,EAAE,CAAC,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,QAAQ,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;CAC3E,CA2cA;AAmCD,iEAAiE;AACjE,MAAM,MAAM,iBAAiB,GAAG,SAAS,GAAG,gBAAgB,GAAG,QAAQ,CAAC;AAOxE;;;;;;GAMG;AACH,wBAAsB,wBAAwB,CAAC,MAAM,EAAE,MAAM,EAAE,IAAI,SAAY,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAsB3G;AAED;;;;;;;;;;;;;GAaG;AACH,wBAAsB,0BAA0B,CAC9C,IAAI,SAAY,EAChB,kBAAkB,GAAE,CAAC,MAAM,EAAE,MAAM,KAAK,MAAM,GAAG,IAAoB,GACpE,OAAO,CAAC,IAAI,CAAC,CA4Bf;AAED;;;;;;GAMG;AACH,wBAAsB,YAAY,CAAC,IAAI,SAAY,GAAG,OAAO,CAAC,OAAO,CAAC,CAgBrE;AAyBD;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAsB,iBAAiB,CAAC,IAAI,SAAY,EAAE,cAAc,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC,CA4ClG;AAoBD;;;;;;;;GAQG;AACH,wBAAsB,yBAAyB,CAAC,MAAM,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,EAAE,kBAAkB,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CA4C/H"}