@dollhousemcp/mcp-server 2.0.18 → 2.0.20
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 +8 -0
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/handlers/ElementCRUDHandler.d.ts +30 -0
- package/dist/handlers/ElementCRUDHandler.d.ts.map +1 -1
- package/dist/handlers/ElementCRUDHandler.js +141 -2
- package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts +3 -0
- package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/MCPAQLHandler.js +98 -1
- package/dist/handlers/mcp-aql/OperationRouter.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/OperationRouter.js +6 -1
- package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/OperationSchema.js +17 -1
- package/dist/handlers/mcp-aql/policies/AgentToolPolicyTranslator.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/policies/AgentToolPolicyTranslator.js +2 -1
- package/dist/handlers/mcp-aql/policies/ElementPolicies.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/policies/ElementPolicies.js +2 -1
- package/dist/handlers/mcp-aql/policies/OperationPolicies.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/policies/OperationPolicies.js +6 -1
- package/dist/handlers/mcp-aql/policies/ToolClassification.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/policies/ToolClassification.js +2 -1
- package/dist/server/tools/MCPAQLTools.js +2 -1
- package/dist/web/console/IngestRoutes.d.ts +6 -0
- package/dist/web/console/IngestRoutes.d.ts.map +1 -1
- package/dist/web/console/IngestRoutes.js +38 -9
- package/dist/web/console/LeaderElection.d.ts +39 -0
- package/dist/web/console/LeaderElection.d.ts.map +1 -1
- package/dist/web/console/LeaderElection.js +147 -29
- package/dist/web/console/LeaderForwardingSink.d.ts.map +1 -1
- package/dist/web/console/LeaderForwardingSink.js +5 -1
- package/dist/web/console/PromotionManager.d.ts.map +1 -1
- package/dist/web/console/PromotionManager.js +3 -11
- package/dist/web/console/StaleProcessRecovery.d.ts +11 -0
- package/dist/web/console/StaleProcessRecovery.d.ts.map +1 -1
- package/dist/web/console/StaleProcessRecovery.js +229 -63
- package/dist/web/console/UnifiedConsole.d.ts +22 -1
- package/dist/web/console/UnifiedConsole.d.ts.map +1 -1
- package/dist/web/console/UnifiedConsole.js +172 -11
- package/dist/web/public/app.js +62 -1
- package/dist/web/public/index.html +19 -17
- package/dist/web/public/sessions.js +111 -0
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +12 -10
- package/package.json +1 -1
- package/server.json +2 -2
|
@@ -12,12 +12,14 @@
|
|
|
12
12
|
*
|
|
13
13
|
* @since v2.1.0 — Issue #1700
|
|
14
14
|
*/
|
|
15
|
+
import { UnicodeValidator } from '../../security/validators/unicodeValidator.js';
|
|
15
16
|
import { logger } from '../../utils/logger.js';
|
|
16
|
-
import { electLeader, isLeaderWebConsoleReachable, forceClaimLeadership, startHeartbeat, registerLeaderCleanup, detectLegacyLeader, } from './LeaderElection.js';
|
|
17
|
+
import { electLeader, isLeaderWebConsoleReachable, forceClaimLeadership, startHeartbeat, registerLeaderCleanup, detectLegacyLeader, readLeaderLock, deleteLeaderLock, LOCK_VERSION, CONSOLE_PROTOCOL_VERSION, LEGACY_SERVER_VERSION, } from './LeaderElection.js';
|
|
17
18
|
import { createIngestRoutes } from './IngestRoutes.js';
|
|
18
19
|
import { LeaderForwardingLogSink, SessionHeartbeat, } from './LeaderForwardingSink.js';
|
|
19
20
|
import { PromotionManager } from './PromotionManager.js';
|
|
20
21
|
import { ConsoleTokenStore } from './consoleToken.js';
|
|
22
|
+
import { findPidOnPort } from './StaleProcessRecovery.js';
|
|
21
23
|
import { env } from '../../config/env.js';
|
|
22
24
|
/**
|
|
23
25
|
* Default console port from the env var. Used as fallback when no port
|
|
@@ -27,6 +29,11 @@ import { env } from '../../config/env.js';
|
|
|
27
29
|
* 3. 41715 (hardcoded default in env.ts)
|
|
28
30
|
*/
|
|
29
31
|
const DEFAULT_CONSOLE_PORT = env.DOLLHOUSE_WEB_CONSOLE_PORT;
|
|
32
|
+
const LEGACY_CONSOLE_FALLBACK_PORT = 3939;
|
|
33
|
+
const SYNTHETIC_PORT_OWNER_SESSION_PREFIX = 'port-owner-';
|
|
34
|
+
function currentTimestamp() {
|
|
35
|
+
return new Date().toISOString();
|
|
36
|
+
}
|
|
30
37
|
/**
|
|
31
38
|
* Check for a running legacy (pre-authentication) DollhouseMCP console and
|
|
32
39
|
* log a WARN-level message if one is found (#1794).
|
|
@@ -56,7 +63,7 @@ export async function warnIfLegacyConsolePresent(currentPort, detect = detectLeg
|
|
|
56
63
|
`(pid=${legacy.pid}, port=${legacy.port}). Both consoles will run ` +
|
|
57
64
|
`independently on different ports with different security posture. ` +
|
|
58
65
|
`The authenticated console (this process) uses port ${currentPort}; ` +
|
|
59
|
-
`the legacy console uses port ${legacy.port ??
|
|
66
|
+
`the legacy console uses port ${legacy.port ?? LEGACY_CONSOLE_FALLBACK_PORT}. ` +
|
|
60
67
|
`For consistent security, update the legacy installation to a ` +
|
|
61
68
|
`version with the authenticated console.`);
|
|
62
69
|
}
|
|
@@ -70,6 +77,135 @@ export async function warnIfLegacyConsolePresent(currentPort, detect = detectLeg
|
|
|
70
77
|
return null;
|
|
71
78
|
}
|
|
72
79
|
}
|
|
80
|
+
function buildDiscoveryHeaders(authToken) {
|
|
81
|
+
return authToken ? { Authorization: `Bearer ${authToken}` } : {};
|
|
82
|
+
}
|
|
83
|
+
function buildLeaderInfoFromSession(port, ownerPid, leaderSession) {
|
|
84
|
+
return {
|
|
85
|
+
version: LOCK_VERSION,
|
|
86
|
+
pid: ownerPid,
|
|
87
|
+
port,
|
|
88
|
+
sessionId: UnicodeValidator.normalize(leaderSession.sessionId).normalizedContent,
|
|
89
|
+
startedAt: leaderSession.startedAt ?? currentTimestamp(),
|
|
90
|
+
heartbeat: leaderSession.lastHeartbeat ?? currentTimestamp(),
|
|
91
|
+
serverVersion: leaderSession.serverVersion ?? LEGACY_SERVER_VERSION,
|
|
92
|
+
consoleProtocolVersion: leaderSession.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
function buildSyntheticLeaderInfo(port, ownerPid) {
|
|
96
|
+
const now = currentTimestamp();
|
|
97
|
+
return {
|
|
98
|
+
version: LOCK_VERSION,
|
|
99
|
+
pid: ownerPid,
|
|
100
|
+
port,
|
|
101
|
+
sessionId: `${SYNTHETIC_PORT_OWNER_SESSION_PREFIX}${ownerPid}`,
|
|
102
|
+
startedAt: now,
|
|
103
|
+
heartbeat: now,
|
|
104
|
+
serverVersion: LEGACY_SERVER_VERSION,
|
|
105
|
+
consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,
|
|
106
|
+
};
|
|
107
|
+
}
|
|
108
|
+
async function discoverLeaderViaSessionsApi(port, ownerPid, authToken, fetchImpl) {
|
|
109
|
+
const response = await fetchImpl(`http://127.0.0.1:${port}/api/sessions`, {
|
|
110
|
+
headers: buildDiscoveryHeaders(authToken),
|
|
111
|
+
});
|
|
112
|
+
if (!response.ok) {
|
|
113
|
+
return null;
|
|
114
|
+
}
|
|
115
|
+
const payload = await response.json();
|
|
116
|
+
const sessions = Array.isArray(payload.sessions) ? payload.sessions : [];
|
|
117
|
+
const leaderSession = sessions.find((session) => session.pid === ownerPid &&
|
|
118
|
+
session.isLeader === true &&
|
|
119
|
+
session.kind === 'mcp' &&
|
|
120
|
+
session.status !== 'stopped');
|
|
121
|
+
return leaderSession ? buildLeaderInfoFromSession(port, ownerPid, leaderSession) : null;
|
|
122
|
+
}
|
|
123
|
+
export async function discoverLeaderServingPort(port, authToken, deps = {}) {
|
|
124
|
+
const fetchImpl = deps.fetchImpl ?? fetch;
|
|
125
|
+
const findPidOnPortImpl = deps.findPidOnPortImpl ?? findPidOnPort;
|
|
126
|
+
const readLeaderLockImpl = deps.readLeaderLockImpl ?? readLeaderLock;
|
|
127
|
+
const ownerPid = await findPidOnPortImpl(port);
|
|
128
|
+
if (ownerPid !== null) {
|
|
129
|
+
try {
|
|
130
|
+
const leaderInfo = await discoverLeaderViaSessionsApi(port, ownerPid, authToken, fetchImpl);
|
|
131
|
+
if (leaderInfo) {
|
|
132
|
+
return { ownerPid, source: 'api', leaderInfo };
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
catch (err) {
|
|
136
|
+
logger.debug('[UnifiedConsole] Failed to query active leader sessions', {
|
|
137
|
+
port,
|
|
138
|
+
ownerPid,
|
|
139
|
+
error: err instanceof Error ? err.message : String(err),
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
const lock = await readLeaderLockImpl();
|
|
144
|
+
if (lock?.port === port && (ownerPid === null || lock.pid === ownerPid)) {
|
|
145
|
+
return {
|
|
146
|
+
ownerPid: ownerPid ?? lock.pid,
|
|
147
|
+
source: 'lock',
|
|
148
|
+
leaderInfo: {
|
|
149
|
+
...lock,
|
|
150
|
+
sessionId: UnicodeValidator.normalize(lock.sessionId).normalizedContent,
|
|
151
|
+
},
|
|
152
|
+
};
|
|
153
|
+
}
|
|
154
|
+
if (ownerPid !== null) {
|
|
155
|
+
return {
|
|
156
|
+
ownerPid,
|
|
157
|
+
source: 'synthetic',
|
|
158
|
+
leaderInfo: buildSyntheticLeaderInfo(port, ownerPid),
|
|
159
|
+
};
|
|
160
|
+
}
|
|
161
|
+
return { leaderInfo: null, ownerPid: null, source: 'none' };
|
|
162
|
+
}
|
|
163
|
+
export async function recoverLeaderBindFailure(provisionalLeader, port, authToken, deps = {}) {
|
|
164
|
+
const readLeaderLockImpl = deps.readLeaderLockImpl ?? readLeaderLock;
|
|
165
|
+
const deleteLeaderLockImpl = deps.deleteLeaderLockImpl ?? deleteLeaderLock;
|
|
166
|
+
logger.info('[UnifiedConsole] Leader bind recovery initiated', {
|
|
167
|
+
provisionalSessionId: provisionalLeader.sessionId,
|
|
168
|
+
provisionalPid: provisionalLeader.pid,
|
|
169
|
+
port,
|
|
170
|
+
});
|
|
171
|
+
let fallback = await discoverLeaderServingPort(port, authToken, deps);
|
|
172
|
+
let lockCleanupAttempted = false;
|
|
173
|
+
let lockCleanupPerformed = false;
|
|
174
|
+
const currentLock = await readLeaderLockImpl();
|
|
175
|
+
const provisionalLockMatches = (currentLock?.pid === provisionalLeader.pid &&
|
|
176
|
+
currentLock.port === provisionalLeader.port &&
|
|
177
|
+
currentLock.sessionId === provisionalLeader.sessionId);
|
|
178
|
+
const fallbackPointsToProvisionalLeader = (fallback.leaderInfo?.pid === provisionalLeader.pid &&
|
|
179
|
+
fallback.leaderInfo.port === provisionalLeader.port &&
|
|
180
|
+
fallback.leaderInfo.sessionId === provisionalLeader.sessionId);
|
|
181
|
+
if (provisionalLockMatches) {
|
|
182
|
+
lockCleanupAttempted = true;
|
|
183
|
+
await deleteLeaderLockImpl();
|
|
184
|
+
lockCleanupPerformed = true;
|
|
185
|
+
logger.info('[UnifiedConsole] Removed provisional leader lock after bind failure', {
|
|
186
|
+
provisionalSessionId: provisionalLeader.sessionId,
|
|
187
|
+
provisionalPid: provisionalLeader.pid,
|
|
188
|
+
port,
|
|
189
|
+
});
|
|
190
|
+
if (fallbackPointsToProvisionalLeader) {
|
|
191
|
+
fallback = await discoverLeaderServingPort(port, authToken, deps);
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
logger.info('[UnifiedConsole] Leader bind recovery completed', {
|
|
195
|
+
provisionalSessionId: provisionalLeader.sessionId,
|
|
196
|
+
provisionalPid: provisionalLeader.pid,
|
|
197
|
+
port,
|
|
198
|
+
discoverySource: fallback.source,
|
|
199
|
+
ownerPid: fallback.ownerPid,
|
|
200
|
+
lockCleanupAttempted,
|
|
201
|
+
lockCleanupPerformed,
|
|
202
|
+
});
|
|
203
|
+
return {
|
|
204
|
+
...fallback,
|
|
205
|
+
lockCleanupAttempted,
|
|
206
|
+
lockCleanupPerformed,
|
|
207
|
+
};
|
|
208
|
+
}
|
|
73
209
|
/**
|
|
74
210
|
* Start the unified web console.
|
|
75
211
|
*
|
|
@@ -134,10 +270,6 @@ async function startAsLeader(options, election, consolePort = DEFAULT_CONSOLE_PO
|
|
|
134
270
|
metricsOnSnapshot: (snapshot) => liveMetricsOnSnapshot?.(snapshot),
|
|
135
271
|
storeMetricsSnapshot: (snapshot) => options.metricsSink?.onSnapshot(snapshot),
|
|
136
272
|
});
|
|
137
|
-
// Register the leader as a session
|
|
138
|
-
ingestResult.registerLeaderSession(options.sessionId, process.pid);
|
|
139
|
-
// Register the web console itself so the session indicator is never empty (#1805)
|
|
140
|
-
ingestResult.registerConsoleSession();
|
|
141
273
|
// Start the web server with ingest routes mounted before the SPA fallback.
|
|
142
274
|
// If the port is occupied by a stale process, retry with exponential backoff.
|
|
143
275
|
const serverOpts = {
|
|
@@ -153,8 +285,34 @@ async function startAsLeader(options, election, consolePort = DEFAULT_CONSOLE_PO
|
|
|
153
285
|
// process on the port, then retrying. No external retry loop needed.
|
|
154
286
|
const webResult = await startWebServer(serverOpts);
|
|
155
287
|
if (webResult.bindResult && !webResult.bindResult.success) {
|
|
156
|
-
|
|
288
|
+
const fallback = await recoverLeaderBindFailure(election.leaderInfo, consolePort, primaryToken.token);
|
|
289
|
+
if (fallback.leaderInfo) {
|
|
290
|
+
logger.warn('[UnifiedConsole] Leader role aborted: bind failed, falling back to follower', {
|
|
291
|
+
port: consolePort,
|
|
292
|
+
bindError: webResult.bindResult.error,
|
|
293
|
+
bindDetail: webResult.bindResult.detail,
|
|
294
|
+
provisionalLeaderPid: election.leaderInfo.pid,
|
|
295
|
+
provisionalLeaderSessionId: election.leaderInfo.sessionId,
|
|
296
|
+
ownerPid: fallback.ownerPid,
|
|
297
|
+
source: fallback.source,
|
|
298
|
+
lockCleanupAttempted: fallback.lockCleanupAttempted,
|
|
299
|
+
lockCleanupPerformed: fallback.lockCleanupPerformed,
|
|
300
|
+
});
|
|
301
|
+
const followerElection = { role: 'follower', leaderInfo: fallback.leaderInfo };
|
|
302
|
+
return startAsFollower(options, followerElection, consolePort, primaryToken.token);
|
|
303
|
+
}
|
|
304
|
+
logger.error('[UnifiedConsole] Leader failed to bind and no active leader could be identified', {
|
|
305
|
+
port: consolePort,
|
|
306
|
+
provisionalLeaderPid: election.leaderInfo.pid,
|
|
307
|
+
bindError: webResult.bindResult.error,
|
|
308
|
+
bindDetail: webResult.bindResult.detail,
|
|
309
|
+
});
|
|
310
|
+
throw new Error(`Leader failed to bind port ${consolePort} and no active leader was discoverable`);
|
|
157
311
|
}
|
|
312
|
+
// Register the leader only after the HTTP listener is actually serving the port.
|
|
313
|
+
ingestResult.registerLeaderSession(options.sessionId, process.pid);
|
|
314
|
+
// Register the web console itself so the session indicator is never empty (#1805)
|
|
315
|
+
ingestResult.registerConsoleSession();
|
|
158
316
|
// Wire SSE broadcasts for this leader's own events
|
|
159
317
|
options.wireSSEBroadcasts(webResult, options.metricsSink);
|
|
160
318
|
// Now wire the live broadcast functions into the ingest routes
|
|
@@ -191,13 +349,16 @@ async function startAsLeader(options, election, consolePort = DEFAULT_CONSOLE_PO
|
|
|
191
349
|
* Start as a follower.
|
|
192
350
|
* Registers forwarding sinks with the LogManager, starts session heartbeat.
|
|
193
351
|
*/
|
|
194
|
-
async function startAsFollower(options, election, consolePort = DEFAULT_CONSOLE_PORT) {
|
|
352
|
+
async function startAsFollower(options, election, consolePort = DEFAULT_CONSOLE_PORT, initialAuthToken = null) {
|
|
195
353
|
const leaderUrl = `http://127.0.0.1:${election.leaderInfo.port}`;
|
|
196
354
|
// Read the console auth token (#1780) written by the leader. May be null
|
|
197
355
|
// if the file doesn't exist yet — the sinks handle that gracefully and
|
|
198
356
|
// simply omit the Bearer header, which is fine when auth is not enforced.
|
|
199
|
-
|
|
200
|
-
|
|
357
|
+
let authToken = initialAuthToken;
|
|
358
|
+
if (authToken === null) {
|
|
359
|
+
const { getPrimaryTokenFromFile } = await import('./consoleToken.js');
|
|
360
|
+
authToken = await getPrimaryTokenFromFile(env.DOLLHOUSE_CONSOLE_TOKEN_FILE);
|
|
361
|
+
}
|
|
201
362
|
if (authToken) {
|
|
202
363
|
logger.debug('[UnifiedConsole] Follower loaded console auth token');
|
|
203
364
|
}
|
|
@@ -233,4 +394,4 @@ async function startAsFollower(options, election, consolePort = DEFAULT_CONSOLE_
|
|
|
233
394
|
},
|
|
234
395
|
};
|
|
235
396
|
}
|
|
236
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UnifiedConsole.js","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAMH,OAAO,EAAE,MAAM,EAAE,MAAM,uBAAuB,CAAC;AAC/C,OAAO,EACL,WAAW,EACX,2BAA2B,EAC3B,oBAAoB,EACpB,cAAc,EACd,qBAAqB,EACrB,kBAAkB,GAEnB,MAAM,qBAAqB,CAAC;AAC7B,OAAO,EAAE,kBAAkB,EAAE,MAAM,mBAAmB,CAAC;AACvD,OAAO,EACL,uBAAuB,EACvB,gBAAgB,GACjB,MAAM,2BAA2B,CAAC;AACnC,OAAO,EAAE,gBAAgB,EAAE,MAAM,uBAAuB,CAAC;AACzD,OAAO,EAAE,iBAAiB,EAAE,MAAM,mBAAmB,CAAC;AACtD,OAAO,EAAE,GAAG,EAAE,MAAM,qBAAqB,CAAC;AAE1C;;;;;;GAMG;AACH,MAAM,oBAAoB,GAAG,GAAG,CAAC,0BAA0B,CAAC;AAoC5D;;;;;;;;;;;;;;;;;;;;GAoBG;AACH,MAAM,CAAC,KAAK,UAAU,0BAA0B,CAC9C,WAAmB,EACnB,SAAoC,kBAAkB,EACtD,MAAqB,MAAM;IAE3B,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,MAAM,MAAM,EAAE,CAAC;QAC9B,IAAI,MAAM,CAAC,aAAa,EAAE,CAAC;YACzB,GAAG,CAAC,IAAI,CACN,6EAA6E;gBAC7E,QAAQ,MAAM,CAAC,GAAG,UAAU,MAAM,CAAC,IAAI,4BAA4B;gBACnE,oEAAoE;gBACpE,sDAAsD,WAAW,IAAI;gBACrE,gCAAgC,MAAM,CAAC,IAAI,IAAI,IAAI,IAAI;gBACvD,+DAA+D;gBAC/D,yCAAyC,CAC1C,CAAC;QACJ,CAAC;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,4DAA4D;QAC5D,GAAG,CAAC,KAAK,CAAC,iDAAiD,EAAE;YAC3D,KAAK,EAAE,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC;SACxD,CAAC,CAAC;QACH,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED;;;;;GAKG;AACH,MAAM,CAAC,KAAK,UAAU,mBAAmB,CAAC,OAA8B;IACtE,0DAA0D;IAC1D,MAAM,WAAW,GAAG,OAAO,CAAC,IAAI,IAAI,oBAAoB,CAAC;IACzD,MAAM,CAAC,KAAK,CAAC,mCAAmC,WAAW,EAAE;QAC3D,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC;IAElE,gEAAgE;IAChE,oEAAoE;IACpE,wEAAwE;IACxE,wEAAwE;IACxE,wCAAwC;IACxC,MAAM,0BAA0B,CAAC,WAAW,CAAC,CAAC;IAE9C,IAAI,QAAQ,GAAG,MAAM,WAAW,CAAC,OAAO,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;IAEjE,kFAAkF;IAClF,qEAAqE;IACrE,oEAAoE;IACpE,IAAI,QAAQ,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;QACjC,MAAM,SAAS,GAAG,MAAM,2BAA2B,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;QACzE,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,QAAQ,GAAG,MAAM,oBAAoB,CAAC,OAAO,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;QACxE,CAAC;IACH,CAAC;IAED,IAAI,QAAQ,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;QAC/B,OAAO,aAAa,CAAC,OAAO,EAAE,QAAQ,EAAE,WAAW,CAAC,CAAC;IACvD,CAAC;SAAM,CAAC;QACN,OAAO,eAAe,CAAC,OAAO,EAAE,QAAQ,EAAE,WAAW,CAAC,CAAC;IACzD,CAAC;AACH,CAAC;AAED;;;;GAIG;AACH,KAAK,UAAU,aAAa,CAC1B,OAA8B,EAC9B,QAAwB,EACxB,cAAsB,oBAAoB;IAE1C,MAAM,EAAE,cAAc,EAAE,GAAG,MAAM,MAAM,CAAC,cAAc,CAAC,CAAC;IACxD,MAAM,EAAE,mBAAmB,EAAE,GAAG,MAAM,MAAM,CAAC,mBAAmB,CAAC,CAAC;IAElE,wEAAwE;IACxE,wEAAwE;IACxE,2EAA2E;IAC3E,yEAAyE;IACzE,uEAAuE;IACvE,MAAM,UAAU,GAAG,IAAI,iBAAiB,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAC;IAC3E,MAAM,YAAY,GAAG,MAAM,UAAU,CAAC,iBAAiB,CAAC,mBAAmB,EAAE,CAAC,CAAC;IAC/E,MAAM,CAAC,IAAI,CAAC,kDAAkD,EAAE;QAC9D,OAAO,EAAE,YAAY,CAAC,EAAE;QACxB,SAAS,EAAE,YAAY,CAAC,IAAI;QAC5B,IAAI,EAAE,UAAU,CAAC,WAAW,EAAE;QAC9B,YAAY,EAAE,GAAG,CAAC,0BAA0B;KAC7C,CAAC,CAAC;IAEH,gFAAgF;IAChF,IAAI,aAA6D,CAAC;IAClE,IAAI,qBAAuE,CAAC;IAE5E,gFAAgF;IAChF,MAAM,YAAY,GAAG,kBAAkB,CAAC;QACtC,YAAY,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,aAAa,EAAE,CAAC,KAAK,CAAC;QAC/C,iBAAiB,EAAE,CAAC,QAAQ,EAAE,EAAE,CAAC,qBAAqB,EAAE,CAAC,QAAQ,CAAC;QAClE,oBAAoB,EAAE,CAAC,QAAQ,EAAE,EAAE,CAAC,OAAO,CAAC,WAAW,EAAE,UAAU,CAAC,QAAQ,CAAC;KAC9E,CAAC,CAAC;IAEH,mCAAmC;IACnC,YAAY,CAAC,qBAAqB,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;IAEnE,kFAAkF;IAClF,YAAY,CAAC,sBAAsB,EAAE,CAAC;IAEtC,2EAA2E;IAC3E,8EAA8E;IAC9E,MAAM,UAAU,GAAG;QACjB,YAAY,EAAE,OAAO,CAAC,YAAY;QAClC,UAAU,EAAE,OAAO,CAAC,UAAU;QAC9B,WAAW,EAAE,OAAO,CAAC,WAAW;QAChC,IAAI,EAAE,WAAW;QACjB,iBAAiB,EAAE,CAAC,YAAY,CAAC,MAAM,CAAC;QACxC,UAAU;QACV,GAAG,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC,EAAE,aAAa,EAAE,OAAO,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;KAC3E,CAAC;IACF,wEAAwE;IACxE,qEAAqE;IACrE,MAAM,SAAS,GAAG,MAAM,cAAc,CAAC,UAAU,CAAC,CAAC;IAEnD,IAAI,SAAS,CAAC,UAAU,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;QAC1D,MAAM,CAAC,KAAK,CAAC,+CAA+C,WAAW,wBAAwB,CAAC,CAAC;IACnG,CAAC;IAED,mDAAmD;IACnD,OAAO,CAAC,iBAAiB,CAAC,SAAS,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC;IAE1D,+DAA+D;IAC/D,IAAI,SAAS,CAAC,YAAY,EAAE,CAAC;QAC3B,MAAM,iBAAiB,GAAG,SAAS,CAAC,YAAY,CAAC;QACjD,6CAA6C;QAC7C,aAAa,GAAG,CAAC,KAAsB,EAAE,EAAE;YACzC,MAAM,OAAO,GAAoB;gBAC/B,GAAG,KAAK;gBACR,IAAI,EAAE,EAAE,GAAG,KAAK,CAAC,IAAI,EAAE,UAAU,EAAE,OAAO,CAAC,SAAS,EAAE;aACvD,CAAC;YACF,iBAAiB,CAAC,OAAO,CAAC,CAAC;QAC7B,CAAC,CAAC;IACJ,CAAC;IACD,qBAAqB,GAAG,SAAS,CAAC,iBAAiB,CAAC;IAEpD,MAAM,CAAC,IAAI,CAAC,2CAA2C,CAAC,CAAC;IAEzD,uCAAuC;IACvC,MAAM,aAAa,GAAG,cAAc,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;IAC1D,qBAAqB,EAAE,CAAC;IAExB,MAAM,CAAC,IAAI,CAAC,iCAAiC,EAAE;QAC7C,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,IAAI,EAAE,WAAW,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG;QACjE,IAAI,EAAE,QAAQ,EAAE,YAAY,EAAE,CAAC,kBAAkB,EAAE,qBAAqB,EAAE,qBAAqB,EAAE,eAAe,CAAC;KAClH,CAAC,CAAC;IAEH,OAAO;QACL,IAAI,EAAE,QAAQ;QACd,QAAQ;QACR,IAAI,EAAE,WAAW;QACjB,OAAO,EAAE,KAAK,IAAI,EAAE;YAClB,aAAa,EAAE,CAAC;QAClB,CAAC;KACF,CAAC;AACJ,CAAC;AAED;;;GAGG;AACH,KAAK,UAAU,eAAe,CAC5B,OAA8B,EAC9B,QAAwB,EACxB,cAAsB,oBAAoB;IAE1C,MAAM,SAAS,GAAG,oBAAoB,QAAQ,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;IAEjE,yEAAyE;IACzE,uEAAuE;IACvE,0EAA0E;IAC1E,MAAM,EAAE,uBAAuB,EAAE,GAAG,MAAM,MAAM,CAAC,mBAAmB,CAAC,CAAC;IACtE,MAAM,SAAS,GAAG,MAAM,uBAAuB,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAC;IAClF,IAAI,SAAS,EAAE,CAAC;QACd,MAAM,CAAC,KAAK,CAAC,qDAAqD,CAAC,CAAC;IACtE,CAAC;SAAM,CAAC;QACN,MAAM,CAAC,KAAK,CAAC,6FAA6F,CAAC,CAAC;IAC9G,CAAC;IAED,qEAAqE;IACrE,0EAA0E;IAC1E,MAAM,YAAY,GAAG,IAAI,gBAAgB,CAAC,OAAO,EAAE,WAAW,EAAE,aAAa,EAAE,eAAe,CAAC,CAAC;IAEhG,0EAA0E;IAC1E,0FAA0F;IAC1F,IAAI,gBAAkC,CAAC;IAEvC,qEAAqE;IACrE,MAAM,cAAc,GAAG,IAAI,uBAAuB,CAAC,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,SAAS,EAAE,GAAG,EAAE;QAC/F,YAAY,CAAC,OAAO,CAAC,cAAc,EAAE,gBAAgB,CAAC;aACnD,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,oCAAoC,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9F,CAAC,CAAC,CAAC;IACH,OAAO,CAAC,eAAe,CAAC,cAAc,CAAC,CAAC;IAExC,wCAAwC;IACxC,gBAAgB,GAAG,IAAI,gBAAgB,CAAC,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,GAAG,EAAE,SAAS,CAAC,CAAC;IAC9F,MAAM,gBAAgB,CAAC,KAAK,EAAE,CAAC;IAE/B,MAAM,CAAC,IAAI,CAAC,mCAAmC,EAAE;QAC/C,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,IAAI,EAAE,UAAU;QAChE,aAAa,EAAE,QAAQ,CAAC,UAAU,CAAC,SAAS,EAAE,SAAS,EAAE,QAAQ,CAAC,UAAU,CAAC,GAAG;QAChF,UAAU,EAAE,QAAQ,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS;KAChD,CAAC,CAAC;IAEH,OAAO;QACL,IAAI,EAAE,UAAU;QAChB,QAAQ;QACR,OAAO,EAAE,KAAK,IAAI,EAAE;YAClB,MAAM,gBAAgB,CAAC,IAAI,EAAE,CAAC;YAC9B,MAAM,cAAc,CAAC,KAAK,EAAE,CAAC;QAC/B,CAAC;KACF,CAAC;AACJ,CAAC","sourcesContent":["/**\n * Unified web console orchestrator.\n *\n * Ties together leader election, console startup, follower wiring,\n * and session lifecycle management. This is the main entry point\n * called by the DI container during deferred setup.\n *\n * Flow:\n * 1. Run leader election (read lock file, claim or follow)\n * 2. If leader: start web server on fixed port, mount ingest routes, start heartbeat\n * 3. If follower: register forwarding sinks with LogManager, start session heartbeat\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport type { UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport type { MemoryLogSink } from '../../logging/sinks/MemoryLogSink.js';\nimport type { MemoryMetricsSink } from '../../metrics/sinks/MemoryMetricsSink.js';\nimport { logger } from '../../utils/logger.js';\nimport {\n  electLeader,\n  isLeaderWebConsoleReachable,\n  forceClaimLeadership,\n  startHeartbeat,\n  registerLeaderCleanup,\n  detectLegacyLeader,\n  type ElectionResult,\n} from './LeaderElection.js';\nimport { createIngestRoutes } from './IngestRoutes.js';\nimport {\n  LeaderForwardingLogSink,\n  SessionHeartbeat,\n} from './LeaderForwardingSink.js';\nimport { PromotionManager } from './PromotionManager.js';\nimport { ConsoleTokenStore } from './consoleToken.js';\nimport { env } from '../../config/env.js';\n\n/**\n * Default console port from the env var. Used as fallback when no port\n * is provided via config file or options. The resolution hierarchy is:\n *   1. options.port (from config file, resolved by the DI container)\n *   2. DOLLHOUSE_WEB_CONSOLE_PORT env var\n *   3. 41715 (hardcoded default in env.ts)\n */\nconst DEFAULT_CONSOLE_PORT = env.DOLLHOUSE_WEB_CONSOLE_PORT;\n\n/**\n * Options for starting the unified console.\n */\nexport interface UnifiedConsoleOptions {\n  /** This process's unique session ID */\n  sessionId: string;\n  /** Portfolio base directory (for startWebServer) */\n  portfolioDir: string;\n  /** Log memory sink (for console history) */\n  memorySink: MemoryLogSink;\n  /** Metrics memory sink */\n  metricsSink?: MemoryMetricsSink;\n  /** MCP-AQL handler for permission routes (typed as any to avoid circular imports) */\n  mcpAqlHandler?: any;\n  /** Callback to register a log sink with the LogManager */\n  registerLogSink: (sink: { write(entry: UnifiedLogEntry): void; flush(): Promise<void>; close(): Promise<void> }) => void;\n  /** Callback to wire SSE broadcasts after web server starts */\n  wireSSEBroadcasts: (webResult: { logBroadcast?: (entry: UnifiedLogEntry) => void; metricsOnSnapshot?: (snapshot: MetricSnapshot) => void }, metricsSink?: MemoryMetricsSink) => void;\n  /** Console port override from config file. Falls back to env var if not provided. */\n  port?: number;\n}\n\n/**\n * Result of starting the unified console.\n */\nexport interface UnifiedConsoleResult {\n  role: 'leader' | 'follower';\n  election: ElectionResult;\n  /** Port the console is running on (leader only) */\n  port?: number;\n  /** Cleanup function to call on shutdown */\n  cleanup: () => Promise<void>;\n}\n\n/**\n * Check for a running legacy (pre-authentication) DollhouseMCP console and\n * log a WARN-level message if one is found (#1794).\n *\n * Extracted from `startUnifiedConsole` so the wiring can be integration-\n * tested in isolation without spinning up a full web server and leader\n * election. The implementation is fire-and-forget: detection failures\n * are logged at DEBUG and never propagate, because a failure here must\n * not block leader election of the authenticated console.\n *\n * @param currentPort - The port the authenticated console intends to\n *                      bind to. Used in the warning message to help the\n *                      user tell the two consoles apart.\n * @param detect      - Optional injection point for the detection\n *                      function. Defaults to `detectLegacyLeader`. Tests\n *                      pass a stub.\n * @param log         - Optional injection point for the logger. Defaults\n *                      to the module logger. Tests pass a spy.\n * @returns The legacy leader info from `detect()`, or null if detection\n *          threw. Exposed so tests can assert the full result shape.\n */\nexport async function warnIfLegacyConsolePresent(\n  currentPort: number,\n  detect: typeof detectLegacyLeader = detectLegacyLeader,\n  log: typeof logger = logger,\n): Promise<Awaited<ReturnType<typeof detectLegacyLeader>> | null> {\n  try {\n    const legacy = await detect();\n    if (legacy.legacyRunning) {\n      log.warn(\n        `[UnifiedConsole] Legacy (pre-authentication) DollhouseMCP console detected ` +\n        `(pid=${legacy.pid}, port=${legacy.port}). Both consoles will run ` +\n        `independently on different ports with different security posture. ` +\n        `The authenticated console (this process) uses port ${currentPort}; ` +\n        `the legacy console uses port ${legacy.port ?? 3939}. ` +\n        `For consistent security, update the legacy installation to a ` +\n        `version with the authenticated console.`,\n      );\n    }\n    return legacy;\n  } catch (err) {\n    // Best-effort — never block election on a detection failure\n    log.debug('[UnifiedConsole] Legacy leader detection failed', {\n      error: err instanceof Error ? err.message : String(err),\n    });\n    return null;\n  }\n}\n\n/**\n * Start the unified web console.\n *\n * Runs leader election, then either starts the full console (leader)\n * or sets up event forwarding (follower).\n */\nexport async function startUnifiedConsole(options: UnifiedConsoleOptions): Promise<UnifiedConsoleResult> {\n  // Resolve port: options (config file) → env var → default\n  const consolePort = options.port || DEFAULT_CONSOLE_PORT;\n  logger.debug(`[UnifiedConsole] Port resolved: ${consolePort}` +\n    (options.port ? ' (from config file)' : ` (from env/default)`));\n\n  // Legacy-leader detection (#1794) — warn the user if a pre-auth\n  // DollhouseMCP console is running alongside this authenticated one.\n  // They will coexist fine because of port + lock + token file isolation,\n  // but the user should know both exist so the differing security posture\n  // between them doesn't look like a bug.\n  await warnIfLegacyConsolePresent(consolePort);\n\n  let election = await electLeader(options.sessionId, consolePort);\n\n  // If we lost the election, check if the leader is actually running a web console.\n  // An MCP stdio process may hold leadership but not serve web routes.\n  // In that case, force a takeover so the web console works properly.\n  if (election.role === 'follower') {\n    const reachable = await isLeaderWebConsoleReachable(election.leaderInfo);\n    if (!reachable) {\n      election = await forceClaimLeadership(options.sessionId, consolePort);\n    }\n  }\n\n  if (election.role === 'leader') {\n    return startAsLeader(options, election, consolePort);\n  } else {\n    return startAsFollower(options, election, consolePort);\n  }\n}\n\n/**\n * Start as the console leader.\n * Binds the resolved console port (config file → env var → default),\n * mounts all routes including ingestion, starts heartbeat.\n */\nasync function startAsLeader(\n  options: UnifiedConsoleOptions,\n  election: ElectionResult,\n  consolePort: number = DEFAULT_CONSOLE_PORT,\n): Promise<UnifiedConsoleResult> {\n  const { startWebServer } = await import('../server.js');\n  const { pickRandomTokenName } = await import('./SessionNames.js');\n\n  // Initialize the console token store (#1780). Creates the token file on\n  // first run, reads the existing tokens on subsequent runs. The token is\n  // persistent across restarts — only rotated on explicit request (Phase 2).\n  // Feature flag DOLLHOUSE_WEB_AUTH_ENABLED controls enforcement; the file\n  // is generated regardless so consumers can attach tokens preemptively.\n  const tokenStore = new ConsoleTokenStore(env.DOLLHOUSE_CONSOLE_TOKEN_FILE);\n  const primaryToken = await tokenStore.ensureInitialized(pickRandomTokenName());\n  logger.info('[UnifiedConsole] Console token store initialized', {\n    tokenId: primaryToken.id,\n    tokenName: primaryToken.name,\n    file: tokenStore.getFilePath(),\n    authEnforced: env.DOLLHOUSE_WEB_AUTH_ENABLED,\n  });\n\n  // Pre-create a placeholder broadcast that we'll wire up after the server starts\n  let liveBroadcast: ((entry: UnifiedLogEntry) => void) | undefined;\n  let liveMetricsOnSnapshot: ((snapshot: MetricSnapshot) => void) | undefined;\n\n  // Create ingestion routes with a deferred broadcast (wired after server starts)\n  const ingestResult = createIngestRoutes({\n    logBroadcast: (entry) => liveBroadcast?.(entry),\n    metricsOnSnapshot: (snapshot) => liveMetricsOnSnapshot?.(snapshot),\n    storeMetricsSnapshot: (snapshot) => options.metricsSink?.onSnapshot(snapshot),\n  });\n\n  // Register the leader as a session\n  ingestResult.registerLeaderSession(options.sessionId, process.pid);\n\n  // Register the web console itself so the session indicator is never empty (#1805)\n  ingestResult.registerConsoleSession();\n\n  // Start the web server with ingest routes mounted before the SPA fallback.\n  // If the port is occupied by a stale process, retry with exponential backoff.\n  const serverOpts = {\n    portfolioDir: options.portfolioDir,\n    memorySink: options.memorySink,\n    metricsSink: options.metricsSink,\n    port: consolePort,\n    additionalRouters: [ingestResult.router],\n    tokenStore,\n    ...(options.mcpAqlHandler ? { mcpAqlHandler: options.mcpAqlHandler } : {}),\n  };\n  // bindAndListen now handles EADDRINUSE by finding and killing the stale\n  // process on the port, then retrying. No external retry loop needed.\n  const webResult = await startWebServer(serverOpts);\n\n  if (webResult.bindResult && !webResult.bindResult.success) {\n    logger.error(`[UnifiedConsole] Leader failed to bind port ${consolePort} — console unavailable`);\n  }\n\n  // Wire SSE broadcasts for this leader's own events\n  options.wireSSEBroadcasts(webResult, options.metricsSink);\n\n  // Now wire the live broadcast functions into the ingest routes\n  if (webResult.logBroadcast) {\n    const originalBroadcast = webResult.logBroadcast;\n    // Stamp leader's own entries with session ID\n    liveBroadcast = (entry: UnifiedLogEntry) => {\n      const stamped: UnifiedLogEntry = {\n        ...entry,\n        data: { ...entry.data, _sessionId: options.sessionId },\n      };\n      originalBroadcast(stamped);\n    };\n  }\n  liveMetricsOnSnapshot = webResult.metricsOnSnapshot;\n\n  logger.info('[UnifiedConsole] Ingestion routes mounted');\n\n  // Start heartbeat and register cleanup\n  const stopHeartbeat = startHeartbeat(election.leaderInfo);\n  registerLeaderCleanup();\n\n  logger.info('[UnifiedConsole] Leader started', {\n    sessionId: options.sessionId, port: consolePort, pid: process.pid,\n    role: 'leader', ingestRoutes: ['/api/ingest/logs', '/api/ingest/metrics', '/api/ingest/session', '/api/sessions'],\n  });\n\n  return {\n    role: 'leader',\n    election,\n    port: consolePort,\n    cleanup: async () => {\n      stopHeartbeat();\n    },\n  };\n}\n\n/**\n * Start as a follower.\n * Registers forwarding sinks with the LogManager, starts session heartbeat.\n */\nasync function startAsFollower(\n  options: UnifiedConsoleOptions,\n  election: ElectionResult,\n  consolePort: number = DEFAULT_CONSOLE_PORT,\n): Promise<UnifiedConsoleResult> {\n  const leaderUrl = `http://127.0.0.1:${election.leaderInfo.port}`;\n\n  // Read the console auth token (#1780) written by the leader. May be null\n  // if the file doesn't exist yet — the sinks handle that gracefully and\n  // simply omit the Bearer header, which is fine when auth is not enforced.\n  const { getPrimaryTokenFromFile } = await import('./consoleToken.js');\n  const authToken = await getPrimaryTokenFromFile(env.DOLLHOUSE_CONSOLE_TOKEN_FILE);\n  if (authToken) {\n    logger.debug('[UnifiedConsole] Follower loaded console auth token');\n  } else {\n    logger.debug('[UnifiedConsole] No console auth token file found; follower will POST without Bearer header');\n  }\n\n  // Per-instance promotion manager — tracks its own attempt counter so\n  // multiple followers don't interfere with each other's promotion budgets.\n  const promotionMgr = new PromotionManager(options, consolePort, startAsLeader, startAsFollower);\n\n  // Declare sessionHeartbeat before the sink so the closure can capture it.\n  // Both are initialized before the callback could possibly fire (needs 5+ failed flushes).\n  let sessionHeartbeat: SessionHeartbeat;\n\n  // Register a forwarding log sink with leader-death callback (#1850).\n  const forwardingSink = new LeaderForwardingLogSink(leaderUrl, options.sessionId, authToken, () => {\n    promotionMgr.promote(forwardingSink, sessionHeartbeat)\n      .catch(err => logger.error('[UnifiedConsole] Promotion crashed', { error: String(err) }));\n  });\n  options.registerLogSink(forwardingSink);\n\n  // Start session heartbeat to the leader\n  sessionHeartbeat = new SessionHeartbeat(leaderUrl, options.sessionId, process.pid, authToken);\n  await sessionHeartbeat.start();\n\n  logger.info('[UnifiedConsole] Follower started', {\n    sessionId: options.sessionId, pid: process.pid, role: 'follower',\n    leaderSession: election.leaderInfo.sessionId, leaderPid: election.leaderInfo.pid,\n    leaderPort: election.leaderInfo.port, leaderUrl,\n  });\n\n  return {\n    role: 'follower',\n    election,\n    cleanup: async () => {\n      await sessionHeartbeat.stop();\n      await forwardingSink.close();\n    },\n  };\n}\n"]}
|
|
397
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UnifiedConsole.js","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Unified web console orchestrator.\n *\n * Ties together leader election, console startup, follower wiring,\n * and session lifecycle management. This is the main entry point\n * called by the DI container during deferred setup.\n *\n * Flow:\n * 1. Run leader election (read lock file, claim or follow)\n * 2. If leader: start web server on fixed port, mount ingest routes, start heartbeat\n * 3. If follower: register forwarding sinks with LogManager, start session heartbeat\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport type { UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport type { MemoryLogSink } from '../../logging/sinks/MemoryLogSink.js';\nimport type { MemoryMetricsSink } from '../../metrics/sinks/MemoryMetricsSink.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { logger } from '../../utils/logger.js';\nimport {\n  electLeader,\n  isLeaderWebConsoleReachable,\n  forceClaimLeadership,\n  startHeartbeat,\n  registerLeaderCleanup,\n  detectLegacyLeader,\n  readLeaderLock,\n  deleteLeaderLock,\n  LOCK_VERSION,\n  CONSOLE_PROTOCOL_VERSION,\n  LEGACY_SERVER_VERSION,\n  type ElectionResult,\n  type ConsoleLeaderInfo,\n} from './LeaderElection.js';\nimport { createIngestRoutes } from './IngestRoutes.js';\nimport {\n  LeaderForwardingLogSink,\n  SessionHeartbeat,\n} from './LeaderForwardingSink.js';\nimport { PromotionManager } from './PromotionManager.js';\nimport { ConsoleTokenStore } from './consoleToken.js';\nimport { findPidOnPort } from './StaleProcessRecovery.js';\nimport { env } from '../../config/env.js';\n\n/**\n * Default console port from the env var. Used as fallback when no port\n * is provided via config file or options. The resolution hierarchy is:\n *   1. options.port (from config file, resolved by the DI container)\n *   2. DOLLHOUSE_WEB_CONSOLE_PORT env var\n *   3. 41715 (hardcoded default in env.ts)\n */\nconst DEFAULT_CONSOLE_PORT = env.DOLLHOUSE_WEB_CONSOLE_PORT;\nconst LEGACY_CONSOLE_FALLBACK_PORT = 3939;\nconst SYNTHETIC_PORT_OWNER_SESSION_PREFIX = 'port-owner-';\n\nfunction currentTimestamp(): string {\n  return new Date().toISOString();\n}\n\n/**\n * Options for starting the unified console.\n */\nexport interface UnifiedConsoleOptions {\n  /** This process's unique session ID */\n  sessionId: string;\n  /** Portfolio base directory (for startWebServer) */\n  portfolioDir: string;\n  /** Log memory sink (for console history) */\n  memorySink: MemoryLogSink;\n  /** Metrics memory sink */\n  metricsSink?: MemoryMetricsSink;\n  /** MCP-AQL handler for permission routes (typed as any to avoid circular imports) */\n  mcpAqlHandler?: any;\n  /** Callback to register a log sink with the LogManager */\n  registerLogSink: (sink: { write(entry: UnifiedLogEntry): void; flush(): Promise<void>; close(): Promise<void> }) => void;\n  /** Callback to wire SSE broadcasts after web server starts */\n  wireSSEBroadcasts: (webResult: { logBroadcast?: (entry: UnifiedLogEntry) => void; metricsOnSnapshot?: (snapshot: MetricSnapshot) => void }, metricsSink?: MemoryMetricsSink) => void;\n  /** Console port override from config file. Falls back to env var if not provided. */\n  port?: number;\n}\n\n/**\n * Result of starting the unified console.\n */\nexport interface UnifiedConsoleResult {\n  role: 'leader' | 'follower';\n  election: ElectionResult;\n  /** Port the console is running on (leader only) */\n  port?: number;\n  /** Cleanup function to call on shutdown */\n  cleanup: () => Promise<void>;\n}\n\n/**\n * Check for a running legacy (pre-authentication) DollhouseMCP console and\n * log a WARN-level message if one is found (#1794).\n *\n * Extracted from `startUnifiedConsole` so the wiring can be integration-\n * tested in isolation without spinning up a full web server and leader\n * election. The implementation is fire-and-forget: detection failures\n * are logged at DEBUG and never propagate, because a failure here must\n * not block leader election of the authenticated console.\n *\n * @param currentPort - The port the authenticated console intends to\n *                      bind to. Used in the warning message to help the\n *                      user tell the two consoles apart.\n * @param detect      - Optional injection point for the detection\n *                      function. Defaults to `detectLegacyLeader`. Tests\n *                      pass a stub.\n * @param log         - Optional injection point for the logger. Defaults\n *                      to the module logger. Tests pass a spy.\n * @returns The legacy leader info from `detect()`, or null if detection\n *          threw. Exposed so tests can assert the full result shape.\n */\nexport async function warnIfLegacyConsolePresent(\n  currentPort: number,\n  detect: typeof detectLegacyLeader = detectLegacyLeader,\n  log: typeof logger = logger,\n): Promise<Awaited<ReturnType<typeof detectLegacyLeader>> | null> {\n  try {\n    const legacy = await detect();\n    if (legacy.legacyRunning) {\n      log.warn(\n        `[UnifiedConsole] Legacy (pre-authentication) DollhouseMCP console detected ` +\n        `(pid=${legacy.pid}, port=${legacy.port}). Both consoles will run ` +\n        `independently on different ports with different security posture. ` +\n        `The authenticated console (this process) uses port ${currentPort}; ` +\n        `the legacy console uses port ${legacy.port ?? LEGACY_CONSOLE_FALLBACK_PORT}. ` +\n        `For consistent security, update the legacy installation to a ` +\n        `version with the authenticated console.`,\n      );\n    }\n    return legacy;\n  } catch (err) {\n    // Best-effort — never block election on a detection failure\n    log.debug('[UnifiedConsole] Legacy leader detection failed', {\n      error: err instanceof Error ? err.message : String(err),\n    });\n    return null;\n  }\n}\n\ninterface SessionApiRecord {\n  sessionId: string;\n  pid: number;\n  startedAt?: string;\n  lastHeartbeat?: string;\n  status?: string;\n  isLeader?: boolean;\n  kind?: string;\n  serverVersion?: string;\n  consoleProtocolVersion?: number;\n}\n\nexport interface PortLeaderDiscovery {\n  leaderInfo: ConsoleLeaderInfo | null;\n  ownerPid: number | null;\n  source: 'api' | 'lock' | 'synthetic' | 'none';\n}\n\nexport interface BindFailureRecoveryResult extends PortLeaderDiscovery {\n  lockCleanupAttempted: boolean;\n  lockCleanupPerformed: boolean;\n}\n\ninterface DiscoveryDependencies {\n  fetchImpl?: typeof fetch;\n  findPidOnPortImpl?: typeof findPidOnPort;\n  readLeaderLockImpl?: typeof readLeaderLock;\n}\n\nfunction buildDiscoveryHeaders(authToken: string | null): Record<string, string> {\n  return authToken ? { Authorization: `Bearer ${authToken}` } : {};\n}\n\nfunction buildLeaderInfoFromSession(port: number, ownerPid: number, leaderSession: SessionApiRecord): ConsoleLeaderInfo {\n  return {\n    version: LOCK_VERSION,\n    pid: ownerPid,\n    port,\n    sessionId: UnicodeValidator.normalize(leaderSession.sessionId).normalizedContent,\n    startedAt: leaderSession.startedAt ?? currentTimestamp(),\n    heartbeat: leaderSession.lastHeartbeat ?? currentTimestamp(),\n    serverVersion: leaderSession.serverVersion ?? LEGACY_SERVER_VERSION,\n    consoleProtocolVersion: leaderSession.consoleProtocolVersion ?? CONSOLE_PROTOCOL_VERSION,\n  };\n}\n\nfunction buildSyntheticLeaderInfo(port: number, ownerPid: number): ConsoleLeaderInfo {\n  const now = currentTimestamp();\n  return {\n    version: LOCK_VERSION,\n    pid: ownerPid,\n    port,\n    sessionId: `${SYNTHETIC_PORT_OWNER_SESSION_PREFIX}${ownerPid}`,\n    startedAt: now,\n    heartbeat: now,\n    serverVersion: LEGACY_SERVER_VERSION,\n    consoleProtocolVersion: CONSOLE_PROTOCOL_VERSION,\n  };\n}\n\nasync function discoverLeaderViaSessionsApi(\n  port: number,\n  ownerPid: number,\n  authToken: string | null,\n  fetchImpl: typeof fetch,\n): Promise<ConsoleLeaderInfo | null> {\n  const response = await fetchImpl(`http://127.0.0.1:${port}/api/sessions`, {\n    headers: buildDiscoveryHeaders(authToken),\n  });\n  if (!response.ok) {\n    return null;\n  }\n\n  const payload = await response.json() as { sessions?: SessionApiRecord[] };\n  const sessions = Array.isArray(payload.sessions) ? payload.sessions : [];\n  const leaderSession = sessions.find((session) =>\n    session.pid === ownerPid &&\n    session.isLeader === true &&\n    session.kind === 'mcp' &&\n    session.status !== 'stopped'\n  );\n  return leaderSession ? buildLeaderInfoFromSession(port, ownerPid, leaderSession) : null;\n}\n\nexport async function discoverLeaderServingPort(\n  port: number,\n  authToken: string | null,\n  deps: DiscoveryDependencies = {},\n): Promise<PortLeaderDiscovery> {\n  const fetchImpl = deps.fetchImpl ?? fetch;\n  const findPidOnPortImpl = deps.findPidOnPortImpl ?? findPidOnPort;\n  const readLeaderLockImpl = deps.readLeaderLockImpl ?? readLeaderLock;\n  const ownerPid = await findPidOnPortImpl(port);\n\n  if (ownerPid !== null) {\n    try {\n      const leaderInfo = await discoverLeaderViaSessionsApi(port, ownerPid, authToken, fetchImpl);\n      if (leaderInfo) {\n        return { ownerPid, source: 'api', leaderInfo };\n      }\n    } catch (err) {\n      logger.debug('[UnifiedConsole] Failed to query active leader sessions', {\n        port,\n        ownerPid,\n        error: err instanceof Error ? err.message : String(err),\n      });\n    }\n  }\n\n  const lock = await readLeaderLockImpl();\n  if (lock?.port === port && (ownerPid === null || lock.pid === ownerPid)) {\n    return {\n      ownerPid: ownerPid ?? lock.pid,\n      source: 'lock',\n      leaderInfo: {\n        ...lock,\n        sessionId: UnicodeValidator.normalize(lock.sessionId).normalizedContent,\n      },\n    };\n  }\n\n  if (ownerPid !== null) {\n    return {\n      ownerPid,\n      source: 'synthetic',\n      leaderInfo: buildSyntheticLeaderInfo(port, ownerPid),\n    };\n  }\n\n  return { leaderInfo: null, ownerPid: null, source: 'none' };\n}\n\ninterface BindFailureRecoveryDependencies extends DiscoveryDependencies {\n  deleteLeaderLockImpl?: typeof deleteLeaderLock;\n}\n\nexport async function recoverLeaderBindFailure(\n  provisionalLeader: ConsoleLeaderInfo,\n  port: number,\n  authToken: string | null,\n  deps: BindFailureRecoveryDependencies = {},\n): Promise<BindFailureRecoveryResult> {\n  const readLeaderLockImpl = deps.readLeaderLockImpl ?? readLeaderLock;\n  const deleteLeaderLockImpl = deps.deleteLeaderLockImpl ?? deleteLeaderLock;\n  logger.info('[UnifiedConsole] Leader bind recovery initiated', {\n    provisionalSessionId: provisionalLeader.sessionId,\n    provisionalPid: provisionalLeader.pid,\n    port,\n  });\n\n  let fallback = await discoverLeaderServingPort(port, authToken, deps);\n  let lockCleanupAttempted = false;\n  let lockCleanupPerformed = false;\n  const currentLock = await readLeaderLockImpl();\n  const provisionalLockMatches = (\n    currentLock?.pid === provisionalLeader.pid &&\n    currentLock.port === provisionalLeader.port &&\n    currentLock.sessionId === provisionalLeader.sessionId\n  );\n  const fallbackPointsToProvisionalLeader = (\n    fallback.leaderInfo?.pid === provisionalLeader.pid &&\n    fallback.leaderInfo.port === provisionalLeader.port &&\n    fallback.leaderInfo.sessionId === provisionalLeader.sessionId\n  );\n\n  if (provisionalLockMatches) {\n    lockCleanupAttempted = true;\n    await deleteLeaderLockImpl();\n    lockCleanupPerformed = true;\n    logger.info('[UnifiedConsole] Removed provisional leader lock after bind failure', {\n      provisionalSessionId: provisionalLeader.sessionId,\n      provisionalPid: provisionalLeader.pid,\n      port,\n    });\n    if (fallbackPointsToProvisionalLeader) {\n      fallback = await discoverLeaderServingPort(port, authToken, deps);\n    }\n  }\n\n  logger.info('[UnifiedConsole] Leader bind recovery completed', {\n    provisionalSessionId: provisionalLeader.sessionId,\n    provisionalPid: provisionalLeader.pid,\n    port,\n    discoverySource: fallback.source,\n    ownerPid: fallback.ownerPid,\n    lockCleanupAttempted,\n    lockCleanupPerformed,\n  });\n\n  return {\n    ...fallback,\n    lockCleanupAttempted,\n    lockCleanupPerformed,\n  };\n}\n\n/**\n * Start the unified web console.\n *\n * Runs leader election, then either starts the full console (leader)\n * or sets up event forwarding (follower).\n */\nexport async function startUnifiedConsole(options: UnifiedConsoleOptions): Promise<UnifiedConsoleResult> {\n  // Resolve port: options (config file) → env var → default\n  const consolePort = options.port || DEFAULT_CONSOLE_PORT;\n  logger.debug(`[UnifiedConsole] Port resolved: ${consolePort}` +\n    (options.port ? ' (from config file)' : ` (from env/default)`));\n\n  // Legacy-leader detection (#1794) — warn the user if a pre-auth\n  // DollhouseMCP console is running alongside this authenticated one.\n  // They will coexist fine because of port + lock + token file isolation,\n  // but the user should know both exist so the differing security posture\n  // between them doesn't look like a bug.\n  await warnIfLegacyConsolePresent(consolePort);\n\n  let election = await electLeader(options.sessionId, consolePort);\n\n  // If we lost the election, check if the leader is actually running a web console.\n  // An MCP stdio process may hold leadership but not serve web routes.\n  // In that case, force a takeover so the web console works properly.\n  if (election.role === 'follower') {\n    const reachable = await isLeaderWebConsoleReachable(election.leaderInfo);\n    if (!reachable) {\n      election = await forceClaimLeadership(options.sessionId, consolePort);\n    }\n  }\n\n  if (election.role === 'leader') {\n    return startAsLeader(options, election, consolePort);\n  } else {\n    return startAsFollower(options, election, consolePort);\n  }\n}\n\n/**\n * Start as the console leader.\n * Binds the resolved console port (config file → env var → default),\n * mounts all routes including ingestion, starts heartbeat.\n */\nasync function startAsLeader(\n  options: UnifiedConsoleOptions,\n  election: ElectionResult,\n  consolePort: number = DEFAULT_CONSOLE_PORT,\n): Promise<UnifiedConsoleResult> {\n  const { startWebServer } = await import('../server.js');\n  const { pickRandomTokenName } = await import('./SessionNames.js');\n\n  // Initialize the console token store (#1780). Creates the token file on\n  // first run, reads the existing tokens on subsequent runs. The token is\n  // persistent across restarts — only rotated on explicit request (Phase 2).\n  // Feature flag DOLLHOUSE_WEB_AUTH_ENABLED controls enforcement; the file\n  // is generated regardless so consumers can attach tokens preemptively.\n  const tokenStore = new ConsoleTokenStore(env.DOLLHOUSE_CONSOLE_TOKEN_FILE);\n  const primaryToken = await tokenStore.ensureInitialized(pickRandomTokenName());\n  logger.info('[UnifiedConsole] Console token store initialized', {\n    tokenId: primaryToken.id,\n    tokenName: primaryToken.name,\n    file: tokenStore.getFilePath(),\n    authEnforced: env.DOLLHOUSE_WEB_AUTH_ENABLED,\n  });\n\n  // Pre-create a placeholder broadcast that we'll wire up after the server starts\n  let liveBroadcast: ((entry: UnifiedLogEntry) => void) | undefined;\n  let liveMetricsOnSnapshot: ((snapshot: MetricSnapshot) => void) | undefined;\n\n  // Create ingestion routes with a deferred broadcast (wired after server starts)\n  const ingestResult = createIngestRoutes({\n    logBroadcast: (entry) => liveBroadcast?.(entry),\n    metricsOnSnapshot: (snapshot) => liveMetricsOnSnapshot?.(snapshot),\n    storeMetricsSnapshot: (snapshot) => options.metricsSink?.onSnapshot(snapshot),\n  });\n\n  // Start the web server with ingest routes mounted before the SPA fallback.\n  // If the port is occupied by a stale process, retry with exponential backoff.\n  const serverOpts = {\n    portfolioDir: options.portfolioDir,\n    memorySink: options.memorySink,\n    metricsSink: options.metricsSink,\n    port: consolePort,\n    additionalRouters: [ingestResult.router],\n    tokenStore,\n    ...(options.mcpAqlHandler ? { mcpAqlHandler: options.mcpAqlHandler } : {}),\n  };\n  // bindAndListen now handles EADDRINUSE by finding and killing the stale\n  // process on the port, then retrying. No external retry loop needed.\n  const webResult = await startWebServer(serverOpts);\n\n  if (webResult.bindResult && !webResult.bindResult.success) {\n    const fallback = await recoverLeaderBindFailure(election.leaderInfo, consolePort, primaryToken.token);\n    if (fallback.leaderInfo) {\n      logger.warn('[UnifiedConsole] Leader role aborted: bind failed, falling back to follower', {\n        port: consolePort,\n        bindError: webResult.bindResult.error,\n        bindDetail: webResult.bindResult.detail,\n        provisionalLeaderPid: election.leaderInfo.pid,\n        provisionalLeaderSessionId: election.leaderInfo.sessionId,\n        ownerPid: fallback.ownerPid,\n        source: fallback.source,\n        lockCleanupAttempted: fallback.lockCleanupAttempted,\n        lockCleanupPerformed: fallback.lockCleanupPerformed,\n      });\n      const followerElection: ElectionResult = { role: 'follower', leaderInfo: fallback.leaderInfo };\n      return startAsFollower(options, followerElection, consolePort, primaryToken.token);\n    }\n\n    logger.error('[UnifiedConsole] Leader failed to bind and no active leader could be identified', {\n      port: consolePort,\n      provisionalLeaderPid: election.leaderInfo.pid,\n      bindError: webResult.bindResult.error,\n      bindDetail: webResult.bindResult.detail,\n    });\n    throw new Error(`Leader failed to bind port ${consolePort} and no active leader was discoverable`);\n  }\n\n  // Register the leader only after the HTTP listener is actually serving the port.\n  ingestResult.registerLeaderSession(options.sessionId, process.pid);\n\n  // Register the web console itself so the session indicator is never empty (#1805)\n  ingestResult.registerConsoleSession();\n\n  // Wire SSE broadcasts for this leader's own events\n  options.wireSSEBroadcasts(webResult, options.metricsSink);\n\n  // Now wire the live broadcast functions into the ingest routes\n  if (webResult.logBroadcast) {\n    const originalBroadcast = webResult.logBroadcast;\n    // Stamp leader's own entries with session ID\n    liveBroadcast = (entry: UnifiedLogEntry) => {\n      const stamped: UnifiedLogEntry = {\n        ...entry,\n        data: { ...entry.data, _sessionId: options.sessionId },\n      };\n      originalBroadcast(stamped);\n    };\n  }\n  liveMetricsOnSnapshot = webResult.metricsOnSnapshot;\n\n  logger.info('[UnifiedConsole] Ingestion routes mounted');\n\n  // Start heartbeat and register cleanup\n  const stopHeartbeat = startHeartbeat(election.leaderInfo);\n  registerLeaderCleanup();\n\n  logger.info('[UnifiedConsole] Leader started', {\n    sessionId: options.sessionId, port: consolePort, pid: process.pid,\n    role: 'leader', ingestRoutes: ['/api/ingest/logs', '/api/ingest/metrics', '/api/ingest/session', '/api/sessions'],\n  });\n\n  return {\n    role: 'leader',\n    election,\n    port: consolePort,\n    cleanup: async () => {\n      stopHeartbeat();\n    },\n  };\n}\n\n/**\n * Start as a follower.\n * Registers forwarding sinks with the LogManager, starts session heartbeat.\n */\nasync function startAsFollower(\n  options: UnifiedConsoleOptions,\n  election: ElectionResult,\n  consolePort: number = DEFAULT_CONSOLE_PORT,\n  initialAuthToken: string | null = null,\n): Promise<UnifiedConsoleResult> {\n  const leaderUrl = `http://127.0.0.1:${election.leaderInfo.port}`;\n\n  // Read the console auth token (#1780) written by the leader. May be null\n  // if the file doesn't exist yet — the sinks handle that gracefully and\n  // simply omit the Bearer header, which is fine when auth is not enforced.\n  let authToken = initialAuthToken;\n  if (authToken === null) {\n    const { getPrimaryTokenFromFile } = await import('./consoleToken.js');\n    authToken = await getPrimaryTokenFromFile(env.DOLLHOUSE_CONSOLE_TOKEN_FILE);\n  }\n  if (authToken) {\n    logger.debug('[UnifiedConsole] Follower loaded console auth token');\n  } else {\n    logger.debug('[UnifiedConsole] No console auth token file found; follower will POST without Bearer header');\n  }\n\n  // Per-instance promotion manager — tracks its own attempt counter so\n  // multiple followers don't interfere with each other's promotion budgets.\n  const promotionMgr = new PromotionManager(options, consolePort, startAsLeader, startAsFollower);\n\n  // Declare sessionHeartbeat before the sink so the closure can capture it.\n  // Both are initialized before the callback could possibly fire (needs 5+ failed flushes).\n  let sessionHeartbeat: SessionHeartbeat;\n\n  // Register a forwarding log sink with leader-death callback (#1850).\n  const forwardingSink = new LeaderForwardingLogSink(leaderUrl, options.sessionId, authToken, () => {\n    promotionMgr.promote(forwardingSink, sessionHeartbeat)\n      .catch(err => logger.error('[UnifiedConsole] Promotion crashed', { error: String(err) }));\n  });\n  options.registerLogSink(forwardingSink);\n\n  // Start session heartbeat to the leader\n  sessionHeartbeat = new SessionHeartbeat(leaderUrl, options.sessionId, process.pid, authToken);\n  await sessionHeartbeat.start();\n\n  logger.info('[UnifiedConsole] Follower started', {\n    sessionId: options.sessionId, pid: process.pid, role: 'follower',\n    leaderSession: election.leaderInfo.sessionId, leaderPid: election.leaderInfo.pid,\n    leaderPort: election.leaderInfo.port, leaderUrl,\n  });\n\n  return {\n    role: 'follower',\n    election,\n    cleanup: async () => {\n      await sessionHeartbeat.stop();\n      await forwardingSink.close();\n    },\n  };\n}\n"]}
|
package/dist/web/public/app.js
CHANGED
|
@@ -1978,12 +1978,70 @@ globalThis.DollhouseConsoleUI.clearBanner = function(bannerId) {
|
|
|
1978
1978
|
|
|
1979
1979
|
const TAB_KEY = 'dollhousemcp-active-tab';
|
|
1980
1980
|
const SETUP_SEEN_KEY = 'dollhousemcp-setup-seen';
|
|
1981
|
+
const FORCED_RELOAD_KEY = 'dollhousemcp-last-forced-reload';
|
|
1981
1982
|
// Server version injected at request time — used to show Setup tab once per version
|
|
1982
1983
|
// so upgraders automatically see it on each new release (not just first-ever visit).
|
|
1983
1984
|
// Validate format (semver-like) before trusting the value; malformed falls back to
|
|
1984
1985
|
// 'unknown' which safely triggers setup on every load rather than silently skipping.
|
|
1985
1986
|
const _rawVersion = document.querySelector('meta[name="dollhouse-server-version"]')?.content || '';
|
|
1986
1987
|
const currentServerVersion = /^\d+\.\d+\.\d+/.test(_rawVersion) ? _rawVersion : 'unknown';
|
|
1988
|
+
const _rawAssetVersion = document.querySelector('meta[name="dollhouse-console-asset-version"]')?.content || '';
|
|
1989
|
+
const currentAssetVersion = /^\d+\.\d+\.\d+/.test(_rawAssetVersion) ? _rawAssetVersion : currentServerVersion;
|
|
1990
|
+
let forcedReloadInFlight = false;
|
|
1991
|
+
|
|
1992
|
+
function normalizeReloadVersion(version) {
|
|
1993
|
+
return typeof version === 'string' && /^\d+\.\d+\.\d+/.test(version)
|
|
1994
|
+
? version
|
|
1995
|
+
: (currentAssetVersion || currentServerVersion || 'unknown');
|
|
1996
|
+
}
|
|
1997
|
+
|
|
1998
|
+
function shouldThrottleForcedReload(targetVersion) {
|
|
1999
|
+
try {
|
|
2000
|
+
const raw = sessionStorage.getItem(FORCED_RELOAD_KEY);
|
|
2001
|
+
if (!raw) return false;
|
|
2002
|
+
const parsed = JSON.parse(raw);
|
|
2003
|
+
return parsed
|
|
2004
|
+
&& parsed.version === targetVersion
|
|
2005
|
+
&& typeof parsed.at === 'number'
|
|
2006
|
+
&& Date.now() - parsed.at < 60_000;
|
|
2007
|
+
} catch {
|
|
2008
|
+
return false;
|
|
2009
|
+
}
|
|
2010
|
+
}
|
|
2011
|
+
|
|
2012
|
+
function rememberForcedReload(targetVersion, reason) {
|
|
2013
|
+
try {
|
|
2014
|
+
sessionStorage.setItem(FORCED_RELOAD_KEY, JSON.stringify({
|
|
2015
|
+
version: targetVersion,
|
|
2016
|
+
reason: reason || 'manual',
|
|
2017
|
+
at: Date.now(),
|
|
2018
|
+
}));
|
|
2019
|
+
} catch {
|
|
2020
|
+
// Ignore storage failures — reload still proceeds.
|
|
2021
|
+
}
|
|
2022
|
+
}
|
|
2023
|
+
|
|
2024
|
+
function buildCacheBustedConsoleUrl(targetVersion, reason) {
|
|
2025
|
+
const url = new URL(globalThis.location.href);
|
|
2026
|
+
url.searchParams.set('dollhouse_bust', targetVersion + '-' + Date.now());
|
|
2027
|
+
url.searchParams.set('dollhouse_asset_version', targetVersion);
|
|
2028
|
+
if (reason) {
|
|
2029
|
+
url.searchParams.set('dollhouse_reload_reason', reason);
|
|
2030
|
+
}
|
|
2031
|
+
return url.toString();
|
|
2032
|
+
}
|
|
2033
|
+
|
|
2034
|
+
function forceConsoleReload(reason, targetVersion) {
|
|
2035
|
+
const normalizedTargetVersion = normalizeReloadVersion(targetVersion);
|
|
2036
|
+
if (forcedReloadInFlight || shouldThrottleForcedReload(normalizedTargetVersion)) {
|
|
2037
|
+
return false;
|
|
2038
|
+
}
|
|
2039
|
+
forcedReloadInFlight = true;
|
|
2040
|
+
rememberForcedReload(normalizedTargetVersion, reason);
|
|
2041
|
+
const reloadUrl = buildCacheBustedConsoleUrl(normalizedTargetVersion, reason);
|
|
2042
|
+
globalThis.location.replace(reloadUrl);
|
|
2043
|
+
return true;
|
|
2044
|
+
}
|
|
1987
2045
|
|
|
1988
2046
|
// Determine which tab to show on load:
|
|
1989
2047
|
// 1. URL hash (deep link)
|
|
@@ -2022,6 +2080,9 @@ globalThis.DollhouseConsoleUI.clearBanner = function(bannerId) {
|
|
|
2022
2080
|
// Expose for other scripts (logs.js, metrics.js, permissions.js)
|
|
2023
2081
|
globalThis.DollhouseConsole = globalThis.DollhouseConsole || {};
|
|
2024
2082
|
globalThis.DollhouseConsole.getUrlParams = () => getTabAndParams().params;
|
|
2083
|
+
globalThis.DollhouseConsole.currentServerVersion = currentServerVersion;
|
|
2084
|
+
globalThis.DollhouseConsole.currentAssetVersion = currentAssetVersion;
|
|
2085
|
+
globalThis.DollhouseConsole.forceReload = forceConsoleReload;
|
|
2025
2086
|
|
|
2026
2087
|
/**
|
|
2027
2088
|
* Apply URL params to the portfolio tab.
|
|
@@ -2175,7 +2236,7 @@ globalThis.DollhouseConsoleUI.clearBanner = function(bannerId) {
|
|
|
2175
2236
|
toast.innerHTML = 'Console session token changed\u2009\u2014\u2009'
|
|
2176
2237
|
+ '<button style="background:#fff;color:#b91c1c;border:none;padding:6px 16px;'
|
|
2177
2238
|
+ 'border-radius:4px;cursor:pointer;font-weight:600;font-size:14px"'
|
|
2178
|
-
+ ' onclick="
|
|
2239
|
+
+ ' onclick="window.DollhouseConsole.forceReload(\'session-expired\')">Reload</button>';
|
|
2179
2240
|
document.body.appendChild(toast);
|
|
2180
2241
|
});
|
|
2181
2242
|
|
|
@@ -14,14 +14,16 @@
|
|
|
14
14
|
<meta name="dollhouse-console-token" content="{{CONSOLE_TOKEN}}">
|
|
15
15
|
<!-- Server version — injected at request time for version-aware UI behaviour (e.g. setup-seen per version). -->
|
|
16
16
|
<meta name="dollhouse-server-version" content="{{DOLLHOUSE_VERSION}}">
|
|
17
|
-
|
|
18
|
-
<
|
|
19
|
-
<link rel="stylesheet" href="
|
|
20
|
-
<link rel="stylesheet" href="
|
|
21
|
-
<link rel="stylesheet" href="
|
|
22
|
-
<link rel="stylesheet" href="
|
|
23
|
-
<link rel="stylesheet" href="
|
|
24
|
-
<link rel="stylesheet" href="
|
|
17
|
+
<!-- Asset version — injected at request time for cache-busting local CSS/JS/img files. -->
|
|
18
|
+
<meta name="dollhouse-console-asset-version" content="{{DOLLHOUSE_ASSET_VERSION}}">
|
|
19
|
+
<link rel="stylesheet" href="fonts.css?v={{DOLLHOUSE_ASSET_VERSION}}">
|
|
20
|
+
<link rel="stylesheet" href="styles.css?v={{DOLLHOUSE_ASSET_VERSION}}">
|
|
21
|
+
<link rel="stylesheet" href="logs.css?v={{DOLLHOUSE_ASSET_VERSION}}">
|
|
22
|
+
<link rel="stylesheet" href="metrics.css?v={{DOLLHOUSE_ASSET_VERSION}}">
|
|
23
|
+
<link rel="stylesheet" href="permissions.css?v={{DOLLHOUSE_ASSET_VERSION}}">
|
|
24
|
+
<link rel="stylesheet" href="sessions.css?v={{DOLLHOUSE_ASSET_VERSION}}">
|
|
25
|
+
<link rel="stylesheet" href="setup.css?v={{DOLLHOUSE_ASSET_VERSION}}">
|
|
26
|
+
<link rel="stylesheet" href="security.css?v={{DOLLHOUSE_ASSET_VERSION}}">
|
|
25
27
|
<!-- uPlot for metrics time-series charts -->
|
|
26
28
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/uplot@1.6.30/dist/uPlot.min.css" integrity="sha384-IfV0B7MIOYuO95kO9G5ySKPz/85zqFNOAs8iy4tkK5zd9izhJAB8b7lHrwYqqmYE" crossorigin="anonymous">
|
|
27
29
|
</head>
|
|
@@ -31,7 +33,7 @@
|
|
|
31
33
|
|
|
32
34
|
<header class="site-header">
|
|
33
35
|
<div class="header-brand">
|
|
34
|
-
<img src="dollhouse-logo.png" alt="DollhouseMCP" class="header-logo" width="32" height="32">
|
|
36
|
+
<img src="dollhouse-logo.png?v={{DOLLHOUSE_ASSET_VERSION}}" alt="DollhouseMCP" class="header-logo" width="32" height="32">
|
|
35
37
|
<div class="header-brand-text">
|
|
36
38
|
<h1 class="site-title">DollhouseMCP</h1>
|
|
37
39
|
<p class="site-tagline">Management Console</p>
|
|
@@ -598,13 +600,13 @@ npm install @dollhousemcp/mcp-server</code></pre>
|
|
|
598
600
|
<script src="https://cdn.jsdelivr.net/npm/uplot@1.6.30/dist/uPlot.iife.min.js" integrity="sha384-1NEYi76CBpge3gahk4+X4M4JzdOV3WYq84RnByqYdAd5SdvJBTNCPFh/nsoHfN6i" crossorigin="anonymous"></script>
|
|
599
601
|
<!-- Console auth helper must load first — it reads the token meta tag and
|
|
600
602
|
exposes window.DollhouseAuth for all subsequent scripts (#1780). -->
|
|
601
|
-
<script src="consoleAuth.js"></script>
|
|
602
|
-
<script src="setup.js"></script>
|
|
603
|
-
<script src="app.js"></script>
|
|
604
|
-
<script src="logs.js"></script>
|
|
605
|
-
<script src="metrics.js"></script>
|
|
606
|
-
<script src="permissions.js"></script>
|
|
607
|
-
<script src="sessions.js"></script>
|
|
608
|
-
<script src="security.js"></script>
|
|
603
|
+
<script src="consoleAuth.js?v={{DOLLHOUSE_ASSET_VERSION}}"></script>
|
|
604
|
+
<script src="setup.js?v={{DOLLHOUSE_ASSET_VERSION}}"></script>
|
|
605
|
+
<script src="app.js?v={{DOLLHOUSE_ASSET_VERSION}}"></script>
|
|
606
|
+
<script src="logs.js?v={{DOLLHOUSE_ASSET_VERSION}}"></script>
|
|
607
|
+
<script src="metrics.js?v={{DOLLHOUSE_ASSET_VERSION}}"></script>
|
|
608
|
+
<script src="permissions.js?v={{DOLLHOUSE_ASSET_VERSION}}"></script>
|
|
609
|
+
<script src="sessions.js?v={{DOLLHOUSE_ASSET_VERSION}}"></script>
|
|
610
|
+
<script src="security.js?v={{DOLLHOUSE_ASSET_VERSION}}"></script>
|
|
609
611
|
</body>
|
|
610
612
|
</html>
|