@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.
- package/CHANGELOG.md +66 -0
- package/README.md +4 -4
- package/README.md.backup +5 -5
- package/README.npm.md +4 -4
- package/dist/di/Container.d.ts +14 -0
- package/dist/di/Container.d.ts.map +1 -1
- package/dist/di/Container.js +43 -26
- package/dist/elements/agents/AgentManager.d.ts +11 -4
- package/dist/elements/agents/AgentManager.d.ts.map +1 -1
- package/dist/elements/agents/AgentManager.js +38 -11
- package/dist/elements/base/BaseElementManager.d.ts +10 -0
- package/dist/elements/base/BaseElementManager.d.ts.map +1 -1
- package/dist/elements/base/BaseElementManager.js +26 -1
- package/dist/elements/ensembles/EnsembleManager.d.ts +1 -0
- package/dist/elements/ensembles/EnsembleManager.d.ts.map +1 -1
- package/dist/elements/ensembles/EnsembleManager.js +13 -2
- package/dist/elements/templates/Template.d.ts +33 -1
- package/dist/elements/templates/Template.d.ts.map +1 -1
- package/dist/elements/templates/Template.js +74 -15
- package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
- package/dist/elements/templates/TemplateManager.js +8 -1
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.d.ts.map +1 -1
- package/dist/generated/version.js +3 -3
- package/dist/handlers/element-crud/createElement.js +2 -2
- package/dist/handlers/mcp-aql/SchemaDispatcher.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/SchemaDispatcher.js +8 -1
- package/dist/handlers/strategies/EnsembleActivationStrategy.d.ts +3 -0
- package/dist/handlers/strategies/EnsembleActivationStrategy.d.ts.map +1 -1
- package/dist/handlers/strategies/EnsembleActivationStrategy.js +48 -9
- package/dist/index.js +19 -13
- package/dist/portfolio/DefaultElementProvider.d.ts +8 -0
- package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
- package/dist/portfolio/DefaultElementProvider.js +43 -1
- package/dist/security/contentValidator.d.ts +15 -0
- package/dist/security/contentValidator.d.ts.map +1 -1
- package/dist/security/contentValidator.js +40 -2
- package/dist/utils/TemplateRenderer.d.ts +9 -0
- package/dist/utils/TemplateRenderer.d.ts.map +1 -1
- package/dist/utils/TemplateRenderer.js +21 -1
- package/dist/web/console/IngestRoutes.d.ts.map +1 -1
- package/dist/web/console/IngestRoutes.js +193 -59
- package/dist/web/console/SessionNames.d.ts +18 -5
- package/dist/web/console/SessionNames.d.ts.map +1 -1
- package/dist/web/console/SessionNames.js +63 -8
- package/dist/web/console/StaleProcessRecovery.d.ts.map +1 -1
- package/dist/web/console/StaleProcessRecovery.js +5 -4
- package/dist/web/console/UnifiedConsole.js +3 -3
- package/dist/web/console/consoleToken.js +3 -3
- package/dist/web/portDiscovery.d.ts +1 -1
- package/dist/web/portDiscovery.d.ts.map +1 -1
- package/dist/web/portDiscovery.js +2 -2
- package/dist/web/public/app.js +65 -11
- package/dist/web/public/index.html +2 -0
- package/dist/web/public/logs.js +24 -2
- package/dist/web/public/metrics.js +22 -4
- package/dist/web/public/sessions.js +55 -8
- package/dist/web/public/setup.js +11 -2
- package/dist/web/public/styles.css +12 -0
- package/dist/web/routes/permissionRoutes.js +2 -2
- package/dist/web/routes/setupRoutes.d.ts +67 -1
- package/dist/web/routes/setupRoutes.d.ts.map +1 -1
- package/dist/web/routes/setupRoutes.js +298 -6
- package/dist/web/routes.d.ts.map +1 -1
- package/dist/web/routes.js +4 -2
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +14 -5
- package/package.json +5 -3
- package/server.json +2 -2
package/dist/web/public/app.js
CHANGED
|
@@ -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('
|
|
156
|
+
.then(r => r.ok ? r.json() : Promise.reject(new Error('collection request failed')))
|
|
121
157
|
.then(mergeCollectionData)
|
|
122
|
-
.catch(() => {
|
|
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.
|
|
1941
|
-
// 2.
|
|
1942
|
-
// 3.
|
|
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
|
-
|
|
2070
|
-
|
|
2071
|
-
|
|
2072
|
-
|
|
2073
|
-
|
|
2074
|
-
|
|
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">
|
package/dist/web/public/logs.js
CHANGED
|
@@ -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 = () =>
|
|
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 {
|
|
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
|
-
|
|
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.
|
|
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
|
-
|
|
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-
|
|
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)
|
|
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 >
|
|
445
|
+
if (document.getElementById('log-session-filter') || retries > SESSION_FILTER_INJECTION_MAX_RETRIES) {
|
|
399
446
|
clearInterval(tryInject);
|
|
400
447
|
}
|
|
401
|
-
},
|
|
448
|
+
}, SESSION_FILTER_INJECTION_RETRY_INTERVAL);
|
|
402
449
|
}
|
|
403
450
|
|
|
404
451
|
if (document.readyState === 'loading') {
|
package/dist/web/public/setup.js
CHANGED
|
@@ -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
|
-
|
|
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;
|
|
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"}
|