@dollhousemcp/mcp-server 2.0.13 → 2.0.14
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/di/Container.d.ts.map +1 -1
- package/dist/di/Container.js +19 -9
- package/dist/elements/BaseElement.js +2 -2
- package/dist/elements/memories/Memory.d.ts.map +1 -1
- package/dist/elements/memories/Memory.js +3 -3
- package/dist/elements/skills/Skill.d.ts.map +1 -1
- package/dist/elements/skills/Skill.js +4 -4
- package/dist/elements/templates/Template.d.ts.map +1 -1
- package/dist/elements/templates/Template.js +4 -4
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/handlers/ElementCRUDHandler.d.ts +10 -0
- package/dist/handlers/ElementCRUDHandler.d.ts.map +1 -1
- package/dist/handlers/ElementCRUDHandler.js +123 -1
- package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts +1 -0
- package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
- package/dist/handlers/mcp-aql/MCPAQLHandler.js +31 -2
- package/dist/services/ActivationStore.d.ts +20 -0
- package/dist/services/ActivationStore.d.ts.map +1 -1
- package/dist/services/ActivationStore.js +104 -1
- package/dist/web/console/IngestRoutes.d.ts +1 -0
- package/dist/web/console/IngestRoutes.d.ts.map +1 -1
- package/dist/web/console/IngestRoutes.js +4 -1
- package/dist/web/console/UnifiedConsole.js +2 -1
- package/dist/web/public/permissions.css +224 -16
- package/dist/web/public/permissions.js +326 -63
- package/dist/web/public/sessions.js +218 -98
- package/dist/web/public/styles.css +15 -10
- package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
- package/dist/web/routes/permissionRoutes.js +57 -19
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +2 -1
- package/package.json +1 -1
- package/server.json +2 -2
|
@@ -29,6 +29,11 @@ export interface PersistedActivation {
|
|
|
29
29
|
/** ISO-8601 timestamp of when activation was persisted */
|
|
30
30
|
activatedAt: string;
|
|
31
31
|
}
|
|
32
|
+
export interface PersistedActivationStateSnapshot {
|
|
33
|
+
sessionId: string;
|
|
34
|
+
lastUpdated: string;
|
|
35
|
+
activations: Record<string, PersistedActivation[]>;
|
|
36
|
+
}
|
|
32
37
|
/**
|
|
33
38
|
* Per-session activation state persistence.
|
|
34
39
|
*
|
|
@@ -82,6 +87,21 @@ export declare class ActivationStore {
|
|
|
82
87
|
* Get all persisted activations for a given element type.
|
|
83
88
|
*/
|
|
84
89
|
getActivations(elementType: string): PersistedActivation[];
|
|
90
|
+
/**
|
|
91
|
+
* Read persisted activation snapshots from disk for reporting/diagnostics.
|
|
92
|
+
*
|
|
93
|
+
* This intentionally does not mutate the store's in-memory state, and it is
|
|
94
|
+
* safe to call from the web console to inspect other sessions' persisted
|
|
95
|
+
* activations without changing live policy enforcement for the current
|
|
96
|
+
* process.
|
|
97
|
+
*/
|
|
98
|
+
listPersistedActivationStates(sessionId?: string): Promise<PersistedActivationStateSnapshot[]>;
|
|
99
|
+
private getPersistedActivationFilenames;
|
|
100
|
+
private readPersistedActivationState;
|
|
101
|
+
private isPersistedActivationState;
|
|
102
|
+
private normalizePersistedActivations;
|
|
103
|
+
private normalizePersistedActivation;
|
|
104
|
+
private logSnapshotReadError;
|
|
85
105
|
/**
|
|
86
106
|
* Get the session ID this store is scoped to.
|
|
87
107
|
*/
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ActivationStore.d.ts","sourceRoot":"","sources":["../../src/services/ActivationStore.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;GAkBG;AAQH,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,4BAA4B,CAAC;AAGxE;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,wDAAwD;IACxD,IAAI,EAAE,MAAM,CAAC;IACb,iEAAiE;IACjE,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,0DAA0D;IAC1D,WAAW,EAAE,MAAM,CAAC;CACrB;
|
|
1
|
+
{"version":3,"file":"ActivationStore.d.ts","sourceRoot":"","sources":["../../src/services/ActivationStore.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;GAkBG;AAQH,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,4BAA4B,CAAC;AAGxE;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,wDAAwD;IACxD,IAAI,EAAE,MAAM,CAAC;IACb,iEAAiE;IACjE,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,0DAA0D;IAC1D,WAAW,EAAE,MAAM,CAAC;CACrB;AAYD,MAAM,WAAW,gCAAgC;IAC/C,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC,MAAM,EAAE,mBAAmB,EAAE,CAAC,CAAC;CACpD;AA4ED;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,qBAAa,eAAe;IAC1B,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAwB;IAChD,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAS;IAClC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAS;IACnC,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAS;IACrC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAU;IAElC,OAAO,CAAC,KAAK,CAA2B;gBAE5B,OAAO,EAAE,qBAAqB,EAAE,QAAQ,CAAC,EAAE,MAAM;IAU7D;;;;OAIG;IACG,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAoEjC;;OAEG;IACH,gBAAgB,CAAC,WAAW,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,IAAI;IA6B5E;;OAEG;IACH,kBAAkB,CAAC,WAAW,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,IAAI;IAoB3D;;OAEG;IACH,qBAAqB,CAAC,WAAW,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,IAAI;IAI9D;;OAEG;IACH,cAAc,CAAC,WAAW,EAAE,MAAM,GAAG,mBAAmB,EAAE;IAK1D;;;;;;;OAOG;IACG,6BAA6B,CAAC,SAAS,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,gCAAgC,EAAE,CAAC;YA6BtF,+BAA+B;YAS/B,4BAA4B;IAqB1C,OAAO,CAAC,0BAA0B;IAOlC,OAAO,CAAC,6BAA6B;IAiBrC,OAAO,CAAC,4BAA4B;IAqBpC,OAAO,CAAC,oBAAoB;IAW5B;;OAEG;IACH,YAAY,IAAI,MAAM;IAItB;;OAEG;IACH,SAAS,IAAI,OAAO;IAIpB;;OAEG;IACH,QAAQ,IAAI,IAAI;IAOhB;;;;OAIG;IACH,OAAO,CAAC,YAAY;IAcpB;;OAEG;YACW,gBAAgB;IAY9B;;OAEG;YACW,OAAO;IAMrB,OAAO,CAAC,gBAAgB;IASxB,OAAO,CAAC,uBAAuB;IAM/B,OAAO,CAAC,mBAAmB;CAS5B"}
|
|
@@ -254,6 +254,109 @@ export class ActivationStore {
|
|
|
254
254
|
const type = normalizeType(elementType);
|
|
255
255
|
return this.state.activations[type] ? [...this.state.activations[type]] : [];
|
|
256
256
|
}
|
|
257
|
+
/**
|
|
258
|
+
* Read persisted activation snapshots from disk for reporting/diagnostics.
|
|
259
|
+
*
|
|
260
|
+
* This intentionally does not mutate the store's in-memory state, and it is
|
|
261
|
+
* safe to call from the web console to inspect other sessions' persisted
|
|
262
|
+
* activations without changing live policy enforcement for the current
|
|
263
|
+
* process.
|
|
264
|
+
*/
|
|
265
|
+
async listPersistedActivationStates(sessionId) {
|
|
266
|
+
if (!this.enabled) {
|
|
267
|
+
return [];
|
|
268
|
+
}
|
|
269
|
+
const normalizedSessionId = typeof sessionId === 'string' && sessionId.trim()
|
|
270
|
+
? normalizeActivationIdentifier(sessionId)
|
|
271
|
+
: undefined;
|
|
272
|
+
try {
|
|
273
|
+
const filenames = await this.getPersistedActivationFilenames(normalizedSessionId);
|
|
274
|
+
const states = await Promise.all(filenames.map(filename => this.readPersistedActivationState(filename)));
|
|
275
|
+
return states
|
|
276
|
+
.flatMap((state) => (state ? [state] : []))
|
|
277
|
+
.sort((a, b) => a.sessionId.localeCompare(b.sessionId));
|
|
278
|
+
}
|
|
279
|
+
catch (error) {
|
|
280
|
+
if (error.code !== 'ENOENT') {
|
|
281
|
+
logger.debug('[ActivationStore] Failed to enumerate activation snapshots for reporting', {
|
|
282
|
+
stateDir: this.stateDir,
|
|
283
|
+
error: error instanceof Error ? error.message : String(error),
|
|
284
|
+
});
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
return [];
|
|
288
|
+
}
|
|
289
|
+
async getPersistedActivationFilenames(sessionId) {
|
|
290
|
+
if (sessionId) {
|
|
291
|
+
return [`activations-${sessionId}.json`];
|
|
292
|
+
}
|
|
293
|
+
const filenames = await fs.readdir(this.stateDir);
|
|
294
|
+
return filenames.filter(name => /^activations-[^.]+\.json$/u.test(name));
|
|
295
|
+
}
|
|
296
|
+
async readPersistedActivationState(filename) {
|
|
297
|
+
const filePath = path.join(this.stateDir, filename);
|
|
298
|
+
try {
|
|
299
|
+
const content = await this.fileOps.readFile(filePath);
|
|
300
|
+
const data = JSON.parse(content);
|
|
301
|
+
if (!this.isPersistedActivationState(data)) {
|
|
302
|
+
return null;
|
|
303
|
+
}
|
|
304
|
+
return {
|
|
305
|
+
sessionId: data.sessionId,
|
|
306
|
+
lastUpdated: data.lastUpdated,
|
|
307
|
+
activations: this.normalizePersistedActivations(data.activations),
|
|
308
|
+
};
|
|
309
|
+
}
|
|
310
|
+
catch (error) {
|
|
311
|
+
this.logSnapshotReadError(filePath, error);
|
|
312
|
+
return null;
|
|
313
|
+
}
|
|
314
|
+
}
|
|
315
|
+
isPersistedActivationState(data) {
|
|
316
|
+
return data.version === 1
|
|
317
|
+
&& typeof data.sessionId === 'string'
|
|
318
|
+
&& Boolean(data.activations)
|
|
319
|
+
&& typeof data.activations === 'object';
|
|
320
|
+
}
|
|
321
|
+
normalizePersistedActivations(activations) {
|
|
322
|
+
const normalized = {};
|
|
323
|
+
for (const [type, entries] of Object.entries(activations)) {
|
|
324
|
+
if (!ACTIVATABLE_TYPES.has(type) || !Array.isArray(entries)) {
|
|
325
|
+
continue;
|
|
326
|
+
}
|
|
327
|
+
const normalizedEntries = entries.flatMap((entry) => this.normalizePersistedActivation(entry));
|
|
328
|
+
if (normalizedEntries.length > 0) {
|
|
329
|
+
normalized[type] = normalizedEntries;
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
return normalized;
|
|
333
|
+
}
|
|
334
|
+
normalizePersistedActivation(entry) {
|
|
335
|
+
if (!entry || typeof entry.name !== 'string') {
|
|
336
|
+
return [];
|
|
337
|
+
}
|
|
338
|
+
const normalizedName = normalizeActivationIdentifier(entry.name);
|
|
339
|
+
if (!normalizedName) {
|
|
340
|
+
return [];
|
|
341
|
+
}
|
|
342
|
+
const normalizedFilename = typeof entry.filename === 'string'
|
|
343
|
+
? normalizeActivationIdentifier(entry.filename)
|
|
344
|
+
: undefined;
|
|
345
|
+
return [{
|
|
346
|
+
...entry,
|
|
347
|
+
name: normalizedName,
|
|
348
|
+
...(normalizedFilename ? { filename: normalizedFilename } : {}),
|
|
349
|
+
}];
|
|
350
|
+
}
|
|
351
|
+
logSnapshotReadError(filePath, error) {
|
|
352
|
+
if (error.code === 'ENOENT') {
|
|
353
|
+
return;
|
|
354
|
+
}
|
|
355
|
+
logger.debug('[ActivationStore] Skipping unreadable activation snapshot during reporting', {
|
|
356
|
+
filePath,
|
|
357
|
+
error: error instanceof Error ? error.message : String(error),
|
|
358
|
+
});
|
|
359
|
+
}
|
|
257
360
|
/**
|
|
258
361
|
* Get the session ID this store is scoped to.
|
|
259
362
|
*/
|
|
@@ -336,4 +439,4 @@ export class ActivationStore {
|
|
|
336
439
|
return counts;
|
|
337
440
|
}
|
|
338
441
|
}
|
|
339
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ActivationStore.js","sourceRoot":"","sources":["../../src/services/ActivationStore.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Activation Store\n *\n * Persists per-session element activation state to disk.\n * Each MCP session (identified by DOLLHOUSE_SESSION_ID) gets its own\n * activation file so concurrent sessions maintain independent profiles.\n *\n * Follows the DangerZoneEnforcer pattern:\n * - DI-managed singleton with FileOperationsService\n * - initialize() loads from disk (tolerates missing/corrupt files)\n * - In-memory state is the hot path; disk writes are fire-and-forget\n * - atomicWriteFile (write-to-temp + rename) prevents partial reads\n *\n * Forward compatibility: The versioned file format (v1) can evolve to\n * include userId, orgId, and audit fields for multi-user HTTPS mode\n * without breaking existing installations.\n *\n * @since v2.0.0 - Issue #598\n */\n\nimport { randomBytes } from 'node:crypto';\nimport os from 'os';\nimport path from 'path';\nimport fs from 'fs/promises';\nimport { logger } from '../utils/logger.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport type { FileOperationsService } from './FileOperationsService.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\n\n/**\n * A persisted activation record for a single element.\n */\nexport interface PersistedActivation {\n  /** Element name (human-readable, used for all types) */\n  name: string;\n  /** For personas only: the filename key used by PersonaManager */\n  filename?: string;\n  /** ISO-8601 timestamp of when activation was persisted */\n  activatedAt: string;\n}\n\n/**\n * Persisted file format (versioned for forward compatibility).\n */\ninterface PersistedActivationState {\n  version: number;\n  sessionId: string;\n  lastUpdated: string;\n  activations: Record<string, PersistedActivation[]>;\n}\n\n/** Valid element types that support activation (stored in singular form) */\nconst ACTIVATABLE_TYPES = new Set(['persona', 'skill', 'agent', 'memory', 'ensemble']);\n\n/**\n * Normalize element type to singular form for consistent storage.\n * ElementType enum uses plural ('personas', 'skills', etc.) but we\n * store in singular form for readability and forward compatibility.\n */\nconst PLURAL_TO_SINGULAR: Record<string, string> = {\n  personas: 'persona',\n  skills: 'skill',\n  agents: 'agent',\n  memories: 'memory',\n  ensembles: 'ensemble',\n};\n\nfunction normalizeType(elementType: string): string {\n  const lower = elementType.toLowerCase();\n  return PLURAL_TO_SINGULAR[lower] ?? lower;\n}\n\nfunction normalizeActivationIdentifier(value: string): string {\n  return UnicodeValidator.normalize(value).normalizedContent.trim();\n}\n\n/** Session ID validation: must start with a letter, then alphanumeric/hyphens/underscores, 1-64 chars */\nconst SESSION_ID_PATTERN = /^[a-zA-Z][a-zA-Z0-9_-]{0,63}$/;\n\n/**\n * Validates and returns the session ID from environment or default.\n *\n * Examples of valid session IDs:\n * - `claude-code` — Claude Code CLI sessions\n * - `zulip-bridge` — Zulip bridge integration\n * - `dev-local` — Local development\n * - `staging-v2` — Staging environment\n */\nfunction resolveSessionId(): string {\n  const envValue = process.env.DOLLHOUSE_SESSION_ID?.trim();\n  if (!envValue) {\n    // Generate a unique session ID per server instance to prevent\n    // cross-session activation leaking (issue #33)\n    const id = `session-${Date.now().toString(36)}-${randomBytes(4).toString('hex')}`;\n    logger.info(`[ActivationStore] No DOLLHOUSE_SESSION_ID set — generated '${id}'`);\n    return id;\n  }\n\n  if (!SESSION_ID_PATTERN.test(envValue)) {\n    logger.warn(\n      `Invalid DOLLHOUSE_SESSION_ID '${envValue}' — must start with a letter, then alphanumeric/hyphens/underscores, 1-64 chars. Falling back to 'default'.`\n    );\n    return 'default';\n  }\n\n  return envValue;\n}\n\n/**\n * Checks whether activation persistence is enabled.\n */\nfunction isPersistenceEnabled(): boolean {\n  const envValue = process.env.DOLLHOUSE_ACTIVATION_PERSISTENCE?.trim().toLowerCase();\n  if (envValue === 'false' || envValue === '0' || envValue === 'no') {\n    return false;\n  }\n  return true;\n}\n\n/** Maximum number of retry attempts for transient disk failures */\nconst PERSIST_MAX_RETRIES = 2;\n\n/** Delay between retry attempts in milliseconds */\nconst PERSIST_RETRY_DELAY_MS = 100;\n\n/**\n * Per-session activation state persistence.\n *\n * Persists element activation state to `~/.dollhouse/state/activations-{sessionId}.json`.\n * Each MCP session is identified by the `DOLLHOUSE_SESSION_ID` environment variable.\n *\n * Thread-safety note: Node.js is single-threaded, so Map operations are safe.\n * For multi-process deployments (future HTTPS), consider Redis or DB backing.\n *\n * @example\n * ```ts\n * // Session ID set via environment:\n * // DOLLHOUSE_SESSION_ID=claude-code\n *\n * const store = new ActivationStore(fileOps);\n * await store.initialize();  // loads ~/.dollhouse/state/activations-claude-code.json\n *\n * store.recordActivation('skill', 'code-reviewer');\n * store.recordActivation('persona', 'Creative Dev', 'creative-dev.md');\n *\n * // On next server start, initialize() restores these activations\n * ```\n */\nexport class ActivationStore {\n  private readonly fileOps: FileOperationsService;\n  private readonly stateDir: string;\n  private readonly sessionId: string;\n  private readonly persistPath: string;\n  private readonly enabled: boolean;\n\n  private state: PersistedActivationState;\n\n  constructor(fileOps: FileOperationsService, stateDir?: string) {\n    this.fileOps = fileOps;\n    this.sessionId = resolveSessionId();\n    this.enabled = isPersistenceEnabled();\n    this.stateDir = stateDir ?? path.join(os.homedir(), '.dollhouse', 'state');\n    this.persistPath = path.join(this.stateDir, `activations-${this.sessionId}.json`);\n\n    this.state = this.createEmptyState();\n  }\n\n  /**\n   * Load persisted activations from disk.\n   * Call once after construction to restore state from a previous session.\n   * If the file is missing or corrupt, starts with empty activations.\n   */\n  async initialize(): Promise<void> {\n    if (!this.enabled) {\n      logger.debug('[ActivationStore] Persistence disabled via DOLLHOUSE_ACTIVATION_PERSISTENCE');\n      return;\n    }\n\n    try {\n      const content = await this.fileOps.readFile(this.persistPath);\n      const data = JSON.parse(content) as PersistedActivationState;\n\n      if (data.version === 1 && data.activations && typeof data.activations === 'object') {\n        // Validate and load only known element types\n        for (const [type, activations] of Object.entries(data.activations)) {\n          if (ACTIVATABLE_TYPES.has(type) && Array.isArray(activations)) {\n            this.state.activations[type] = activations.flatMap((a) => {\n              if (!a || typeof a.name !== 'string') return [];\n\n              const normalizedName = normalizeActivationIdentifier(a.name);\n              if (!normalizedName) return [];\n\n              const normalizedFilename = typeof a.filename === 'string'\n                ? normalizeActivationIdentifier(a.filename)\n                : undefined;\n\n              return [{\n                ...a,\n                name: normalizedName,\n                ...(normalizedFilename ? { filename: normalizedFilename } : {}),\n              }];\n            });\n          }\n        }\n\n        const totalCount = this.getTotalActivationCount();\n        if (totalCount > 0) {\n          logger.info(\n            `[ActivationStore] Restored ${totalCount} activation(s) for session '${this.sessionId}'`\n          );\n\n          SecurityMonitor.logSecurityEvent({\n            type: 'ELEMENT_ACTIVATED',\n            severity: 'LOW',\n            source: 'ActivationStore.initialize',\n            details: `Restored ${totalCount} activation(s) from disk for session '${this.sessionId}'`,\n            additionalData: {\n              sessionId: this.sessionId,\n              counts: this.getActivationCounts(),\n            },\n          });\n        }\n      }\n    } catch (error) {\n      if ((error as NodeJS.ErrnoException).code === 'ENOENT') {\n        logger.debug(`[ActivationStore] No activation file found for session '${this.sessionId}', starting fresh`);\n      } else {\n        logger.warn(`[ActivationStore] Failed to load activation file for session '${this.sessionId}', starting fresh`, { error });\n\n        SecurityMonitor.logSecurityEvent({\n          type: 'ELEMENT_ACTIVATED',\n          severity: 'MEDIUM',\n          source: 'ActivationStore.initialize',\n          details: `Failed to load activation file for session '${this.sessionId}' — starting fresh (possible data corruption)`,\n          additionalData: { error: String(error), sessionId: this.sessionId },\n        });\n      }\n    }\n  }\n\n  /**\n   * Record an element activation. Fire-and-forget persist.\n   */\n  recordActivation(elementType: string, name: string, filename?: string): void {\n    if (!this.enabled) return;\n\n    const type = normalizeType(elementType);\n    if (!ACTIVATABLE_TYPES.has(type)) return;\n    const normalizedName = normalizeActivationIdentifier(name);\n    if (!normalizedName) return;\n    const normalizedFilename = typeof filename === 'string'\n      ? normalizeActivationIdentifier(filename)\n      : undefined;\n\n    if (!this.state.activations[type]) {\n      this.state.activations[type] = [];\n    }\n\n    // Deduplicate — don't add if already present\n    const existing = this.state.activations[type]!;\n    const alreadyActive = existing.some(a => a.name === normalizedName);\n    if (alreadyActive) return;\n\n    existing.push({\n      name: normalizedName,\n      ...(normalizedFilename ? { filename: normalizedFilename } : {}),\n      activatedAt: new Date().toISOString(),\n    });\n\n    this.persistAsync();\n  }\n\n  /**\n   * Record an element deactivation. Fire-and-forget persist.\n   */\n  recordDeactivation(elementType: string, name: string): void {\n    if (!this.enabled) return;\n\n    const type = normalizeType(elementType);\n    if (!ACTIVATABLE_TYPES.has(type)) return;\n    const normalizedName = normalizeActivationIdentifier(name);\n    if (!normalizedName) return;\n\n    const activations = this.state.activations[type];\n    if (!activations) return;\n\n    const initialLength = activations.length;\n    this.state.activations[type] = activations.filter(a => a.name !== normalizedName);\n\n    // Only persist if something actually changed\n    if (this.state.activations[type]!.length !== initialLength) {\n      this.persistAsync();\n    }\n  }\n\n  /**\n   * Remove a specific activation by name (used during restore to prune stale entries).\n   */\n  removeStaleActivation(elementType: string, name: string): void {\n    this.recordDeactivation(elementType, name);\n  }\n\n  /**\n   * Get all persisted activations for a given element type.\n   */\n  getActivations(elementType: string): PersistedActivation[] {\n    const type = normalizeType(elementType);\n    return this.state.activations[type] ? [...this.state.activations[type]!] : [];\n  }\n\n  /**\n   * Get the session ID this store is scoped to.\n   */\n  getSessionId(): string {\n    return this.sessionId;\n  }\n\n  /**\n   * Check if persistence is enabled.\n   */\n  isEnabled(): boolean {\n    return this.enabled;\n  }\n\n  /**\n   * Clear all persisted activations. Used for testing or admin reset.\n   */\n  clearAll(): void {\n    this.state = this.createEmptyState();\n    if (this.enabled) {\n      this.persistAsync();\n    }\n  }\n\n  /**\n   * Fire-and-forget persistence with retry for transient disk failures.\n   * Retries up to PERSIST_MAX_RETRIES times with a short delay.\n   * Disk failure does not block activation operations.\n   */\n  private persistAsync(): void {\n    this.persistWithRetry(0).catch(error => {\n      logger.warn('[ActivationStore] Failed to persist activation state after retries', { error });\n\n      SecurityMonitor.logSecurityEvent({\n        type: 'ELEMENT_ACTIVATED',\n        severity: 'MEDIUM',\n        source: 'ActivationStore.persistAsync',\n        details: `Failed to persist activation state for session '${this.sessionId}' after ${PERSIST_MAX_RETRIES + 1} attempts — state continues in-memory only`,\n        additionalData: { error: String(error), sessionId: this.sessionId },\n      });\n    });\n  }\n\n  /**\n   * Attempt to persist with retries for transient failures (e.g., EBUSY, EAGAIN).\n   */\n  private async persistWithRetry(attempt: number): Promise<void> {\n    try {\n      await this.persist();\n    } catch (error) {\n      if (attempt < PERSIST_MAX_RETRIES) {\n        await new Promise(resolve => setTimeout(resolve, PERSIST_RETRY_DELAY_MS));\n        return this.persistWithRetry(attempt + 1);\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Write current activation state to disk.\n   */\n  private async persist(): Promise<void> {\n    this.state.lastUpdated = new Date().toISOString();\n    await fs.mkdir(this.stateDir, { recursive: true });\n    await this.fileOps.writeFile(this.persistPath, JSON.stringify(this.state, null, 2));\n  }\n\n  private createEmptyState(): PersistedActivationState {\n    return {\n      version: 1,\n      sessionId: this.sessionId,\n      lastUpdated: new Date().toISOString(),\n      activations: {},\n    };\n  }\n\n  private getTotalActivationCount(): number {\n    return Object.values(this.state.activations).reduce(\n      (sum, arr) => sum + (arr?.length ?? 0), 0\n    );\n  }\n\n  private getActivationCounts(): Record<string, number> {\n    const counts: Record<string, number> = {};\n    for (const [type, arr] of Object.entries(this.state.activations)) {\n      if (arr && arr.length > 0) {\n        counts[type] = arr.length;\n      }\n    }\n    return counts;\n  }\n}\n"]}
|
|
442
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ActivationStore.js","sourceRoot":"","sources":["../../src/services/ActivationStore.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Activation Store\n *\n * Persists per-session element activation state to disk.\n * Each MCP session (identified by DOLLHOUSE_SESSION_ID) gets its own\n * activation file so concurrent sessions maintain independent profiles.\n *\n * Follows the DangerZoneEnforcer pattern:\n * - DI-managed singleton with FileOperationsService\n * - initialize() loads from disk (tolerates missing/corrupt files)\n * - In-memory state is the hot path; disk writes are fire-and-forget\n * - atomicWriteFile (write-to-temp + rename) prevents partial reads\n *\n * Forward compatibility: The versioned file format (v1) can evolve to\n * include userId, orgId, and audit fields for multi-user HTTPS mode\n * without breaking existing installations.\n *\n * @since v2.0.0 - Issue #598\n */\n\nimport { randomBytes } from 'node:crypto';\nimport os from 'os';\nimport path from 'path';\nimport fs from 'fs/promises';\nimport { logger } from '../utils/logger.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport type { FileOperationsService } from './FileOperationsService.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\n\n/**\n * A persisted activation record for a single element.\n */\nexport interface PersistedActivation {\n  /** Element name (human-readable, used for all types) */\n  name: string;\n  /** For personas only: the filename key used by PersonaManager */\n  filename?: string;\n  /** ISO-8601 timestamp of when activation was persisted */\n  activatedAt: string;\n}\n\n/**\n * Persisted file format (versioned for forward compatibility).\n */\ninterface PersistedActivationState {\n  version: number;\n  sessionId: string;\n  lastUpdated: string;\n  activations: Record<string, PersistedActivation[]>;\n}\n\nexport interface PersistedActivationStateSnapshot {\n  sessionId: string;\n  lastUpdated: string;\n  activations: Record<string, PersistedActivation[]>;\n}\n\n/** Valid element types that support activation (stored in singular form) */\nconst ACTIVATABLE_TYPES = new Set(['persona', 'skill', 'agent', 'memory', 'ensemble']);\n\n/**\n * Normalize element type to singular form for consistent storage.\n * ElementType enum uses plural ('personas', 'skills', etc.) but we\n * store in singular form for readability and forward compatibility.\n */\nconst PLURAL_TO_SINGULAR: Record<string, string> = {\n  personas: 'persona',\n  skills: 'skill',\n  agents: 'agent',\n  memories: 'memory',\n  ensembles: 'ensemble',\n};\n\nfunction normalizeType(elementType: string): string {\n  const lower = elementType.toLowerCase();\n  return PLURAL_TO_SINGULAR[lower] ?? lower;\n}\n\nfunction normalizeActivationIdentifier(value: string): string {\n  return UnicodeValidator.normalize(value).normalizedContent.trim();\n}\n\n/** Session ID validation: must start with a letter, then alphanumeric/hyphens/underscores, 1-64 chars */\nconst SESSION_ID_PATTERN = /^[a-zA-Z][a-zA-Z0-9_-]{0,63}$/;\n\n/**\n * Validates and returns the session ID from environment or default.\n *\n * Examples of valid session IDs:\n * - `claude-code` — Claude Code CLI sessions\n * - `zulip-bridge` — Zulip bridge integration\n * - `dev-local` — Local development\n * - `staging-v2` — Staging environment\n */\nfunction resolveSessionId(): string {\n  const envValue = process.env.DOLLHOUSE_SESSION_ID?.trim();\n  if (!envValue) {\n    // Generate a unique session ID per server instance to prevent\n    // cross-session activation leaking (issue #33)\n    const id = `session-${Date.now().toString(36)}-${randomBytes(4).toString('hex')}`;\n    logger.info(`[ActivationStore] No DOLLHOUSE_SESSION_ID set — generated '${id}'`);\n    return id;\n  }\n\n  if (!SESSION_ID_PATTERN.test(envValue)) {\n    logger.warn(\n      `Invalid DOLLHOUSE_SESSION_ID '${envValue}' — must start with a letter, then alphanumeric/hyphens/underscores, 1-64 chars. Falling back to 'default'.`\n    );\n    return 'default';\n  }\n\n  return envValue;\n}\n\n/**\n * Checks whether activation persistence is enabled.\n */\nfunction isPersistenceEnabled(): boolean {\n  const envValue = process.env.DOLLHOUSE_ACTIVATION_PERSISTENCE?.trim().toLowerCase();\n  if (envValue === 'false' || envValue === '0' || envValue === 'no') {\n    return false;\n  }\n  return true;\n}\n\n/** Maximum number of retry attempts for transient disk failures */\nconst PERSIST_MAX_RETRIES = 2;\n\n/** Delay between retry attempts in milliseconds */\nconst PERSIST_RETRY_DELAY_MS = 100;\n\n/**\n * Per-session activation state persistence.\n *\n * Persists element activation state to `~/.dollhouse/state/activations-{sessionId}.json`.\n * Each MCP session is identified by the `DOLLHOUSE_SESSION_ID` environment variable.\n *\n * Thread-safety note: Node.js is single-threaded, so Map operations are safe.\n * For multi-process deployments (future HTTPS), consider Redis or DB backing.\n *\n * @example\n * ```ts\n * // Session ID set via environment:\n * // DOLLHOUSE_SESSION_ID=claude-code\n *\n * const store = new ActivationStore(fileOps);\n * await store.initialize();  // loads ~/.dollhouse/state/activations-claude-code.json\n *\n * store.recordActivation('skill', 'code-reviewer');\n * store.recordActivation('persona', 'Creative Dev', 'creative-dev.md');\n *\n * // On next server start, initialize() restores these activations\n * ```\n */\nexport class ActivationStore {\n  private readonly fileOps: FileOperationsService;\n  private readonly stateDir: string;\n  private readonly sessionId: string;\n  private readonly persistPath: string;\n  private readonly enabled: boolean;\n\n  private state: PersistedActivationState;\n\n  constructor(fileOps: FileOperationsService, stateDir?: string) {\n    this.fileOps = fileOps;\n    this.sessionId = resolveSessionId();\n    this.enabled = isPersistenceEnabled();\n    this.stateDir = stateDir ?? path.join(os.homedir(), '.dollhouse', 'state');\n    this.persistPath = path.join(this.stateDir, `activations-${this.sessionId}.json`);\n\n    this.state = this.createEmptyState();\n  }\n\n  /**\n   * Load persisted activations from disk.\n   * Call once after construction to restore state from a previous session.\n   * If the file is missing or corrupt, starts with empty activations.\n   */\n  async initialize(): Promise<void> {\n    if (!this.enabled) {\n      logger.debug('[ActivationStore] Persistence disabled via DOLLHOUSE_ACTIVATION_PERSISTENCE');\n      return;\n    }\n\n    try {\n      const content = await this.fileOps.readFile(this.persistPath);\n      const data = JSON.parse(content) as PersistedActivationState;\n\n      if (data.version === 1 && data.activations && typeof data.activations === 'object') {\n        // Validate and load only known element types\n        for (const [type, activations] of Object.entries(data.activations)) {\n          if (ACTIVATABLE_TYPES.has(type) && Array.isArray(activations)) {\n            this.state.activations[type] = activations.flatMap((a) => {\n              if (!a || typeof a.name !== 'string') return [];\n\n              const normalizedName = normalizeActivationIdentifier(a.name);\n              if (!normalizedName) return [];\n\n              const normalizedFilename = typeof a.filename === 'string'\n                ? normalizeActivationIdentifier(a.filename)\n                : undefined;\n\n              return [{\n                ...a,\n                name: normalizedName,\n                ...(normalizedFilename ? { filename: normalizedFilename } : {}),\n              }];\n            });\n          }\n        }\n\n        const totalCount = this.getTotalActivationCount();\n        if (totalCount > 0) {\n          logger.info(\n            `[ActivationStore] Restored ${totalCount} activation(s) for session '${this.sessionId}'`\n          );\n\n          SecurityMonitor.logSecurityEvent({\n            type: 'ELEMENT_ACTIVATED',\n            severity: 'LOW',\n            source: 'ActivationStore.initialize',\n            details: `Restored ${totalCount} activation(s) from disk for session '${this.sessionId}'`,\n            additionalData: {\n              sessionId: this.sessionId,\n              counts: this.getActivationCounts(),\n            },\n          });\n        }\n      }\n    } catch (error) {\n      if ((error as NodeJS.ErrnoException).code === 'ENOENT') {\n        logger.debug(`[ActivationStore] No activation file found for session '${this.sessionId}', starting fresh`);\n      } else {\n        logger.warn(`[ActivationStore] Failed to load activation file for session '${this.sessionId}', starting fresh`, { error });\n\n        SecurityMonitor.logSecurityEvent({\n          type: 'ELEMENT_ACTIVATED',\n          severity: 'MEDIUM',\n          source: 'ActivationStore.initialize',\n          details: `Failed to load activation file for session '${this.sessionId}' — starting fresh (possible data corruption)`,\n          additionalData: { error: String(error), sessionId: this.sessionId },\n        });\n      }\n    }\n  }\n\n  /**\n   * Record an element activation. Fire-and-forget persist.\n   */\n  recordActivation(elementType: string, name: string, filename?: string): void {\n    if (!this.enabled) return;\n\n    const type = normalizeType(elementType);\n    if (!ACTIVATABLE_TYPES.has(type)) return;\n    const normalizedName = normalizeActivationIdentifier(name);\n    if (!normalizedName) return;\n    const normalizedFilename = typeof filename === 'string'\n      ? normalizeActivationIdentifier(filename)\n      : undefined;\n\n    if (!this.state.activations[type]) {\n      this.state.activations[type] = [];\n    }\n\n    // Deduplicate — don't add if already present\n    const existing = this.state.activations[type]!;\n    const alreadyActive = existing.some(a => a.name === normalizedName);\n    if (alreadyActive) return;\n\n    existing.push({\n      name: normalizedName,\n      ...(normalizedFilename ? { filename: normalizedFilename } : {}),\n      activatedAt: new Date().toISOString(),\n    });\n\n    this.persistAsync();\n  }\n\n  /**\n   * Record an element deactivation. Fire-and-forget persist.\n   */\n  recordDeactivation(elementType: string, name: string): void {\n    if (!this.enabled) return;\n\n    const type = normalizeType(elementType);\n    if (!ACTIVATABLE_TYPES.has(type)) return;\n    const normalizedName = normalizeActivationIdentifier(name);\n    if (!normalizedName) return;\n\n    const activations = this.state.activations[type];\n    if (!activations) return;\n\n    const initialLength = activations.length;\n    this.state.activations[type] = activations.filter(a => a.name !== normalizedName);\n\n    // Only persist if something actually changed\n    if (this.state.activations[type]!.length !== initialLength) {\n      this.persistAsync();\n    }\n  }\n\n  /**\n   * Remove a specific activation by name (used during restore to prune stale entries).\n   */\n  removeStaleActivation(elementType: string, name: string): void {\n    this.recordDeactivation(elementType, name);\n  }\n\n  /**\n   * Get all persisted activations for a given element type.\n   */\n  getActivations(elementType: string): PersistedActivation[] {\n    const type = normalizeType(elementType);\n    return this.state.activations[type] ? [...this.state.activations[type]!] : [];\n  }\n\n  /**\n   * Read persisted activation snapshots from disk for reporting/diagnostics.\n   *\n   * This intentionally does not mutate the store's in-memory state, and it is\n   * safe to call from the web console to inspect other sessions' persisted\n   * activations without changing live policy enforcement for the current\n   * process.\n   */\n  async listPersistedActivationStates(sessionId?: string): Promise<PersistedActivationStateSnapshot[]> {\n    if (!this.enabled) {\n      return [];\n    }\n\n    const normalizedSessionId = typeof sessionId === 'string' && sessionId.trim()\n      ? normalizeActivationIdentifier(sessionId)\n      : undefined;\n\n    try {\n      const filenames = await this.getPersistedActivationFilenames(normalizedSessionId);\n      const states = await Promise.all(\n        filenames.map(filename => this.readPersistedActivationState(filename)),\n      );\n      return states\n        .flatMap((state) => (state ? [state] : []))\n        .sort((a, b) => a.sessionId.localeCompare(b.sessionId));\n    } catch (error) {\n      if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {\n        logger.debug('[ActivationStore] Failed to enumerate activation snapshots for reporting', {\n          stateDir: this.stateDir,\n          error: error instanceof Error ? error.message : String(error),\n        });\n      }\n    }\n\n    return [];\n  }\n\n  private async getPersistedActivationFilenames(sessionId?: string): Promise<string[]> {\n    if (sessionId) {\n      return [`activations-${sessionId}.json`];\n    }\n\n    const filenames = await fs.readdir(this.stateDir);\n    return filenames.filter(name => /^activations-[^.]+\\.json$/u.test(name));\n  }\n\n  private async readPersistedActivationState(filename: string): Promise<PersistedActivationStateSnapshot | null> {\n    const filePath = path.join(this.stateDir, filename);\n\n    try {\n      const content = await this.fileOps.readFile(filePath);\n      const data = JSON.parse(content) as PersistedActivationState;\n      if (!this.isPersistedActivationState(data)) {\n        return null;\n      }\n\n      return {\n        sessionId: data.sessionId,\n        lastUpdated: data.lastUpdated,\n        activations: this.normalizePersistedActivations(data.activations),\n      };\n    } catch (error) {\n      this.logSnapshotReadError(filePath, error);\n      return null;\n    }\n  }\n\n  private isPersistedActivationState(data: PersistedActivationState): boolean {\n    return data.version === 1\n      && typeof data.sessionId === 'string'\n      && Boolean(data.activations)\n      && typeof data.activations === 'object';\n  }\n\n  private normalizePersistedActivations(activations: PersistedActivationState['activations']): Record<string, PersistedActivation[]> {\n    const normalized: Record<string, PersistedActivation[]> = {};\n\n    for (const [type, entries] of Object.entries(activations)) {\n      if (!ACTIVATABLE_TYPES.has(type) || !Array.isArray(entries)) {\n        continue;\n      }\n\n      const normalizedEntries = entries.flatMap((entry) => this.normalizePersistedActivation(entry));\n      if (normalizedEntries.length > 0) {\n        normalized[type] = normalizedEntries;\n      }\n    }\n\n    return normalized;\n  }\n\n  private normalizePersistedActivation(entry: PersistedActivation | null | undefined): PersistedActivation[] {\n    if (!entry || typeof entry.name !== 'string') {\n      return [];\n    }\n\n    const normalizedName = normalizeActivationIdentifier(entry.name);\n    if (!normalizedName) {\n      return [];\n    }\n\n    const normalizedFilename = typeof entry.filename === 'string'\n      ? normalizeActivationIdentifier(entry.filename)\n      : undefined;\n\n    return [{\n      ...entry,\n      name: normalizedName,\n      ...(normalizedFilename ? { filename: normalizedFilename } : {}),\n    }];\n  }\n\n  private logSnapshotReadError(filePath: string, error: unknown): void {\n    if ((error as NodeJS.ErrnoException).code === 'ENOENT') {\n      return;\n    }\n\n    logger.debug('[ActivationStore] Skipping unreadable activation snapshot during reporting', {\n      filePath,\n      error: error instanceof Error ? error.message : String(error),\n    });\n  }\n\n  /**\n   * Get the session ID this store is scoped to.\n   */\n  getSessionId(): string {\n    return this.sessionId;\n  }\n\n  /**\n   * Check if persistence is enabled.\n   */\n  isEnabled(): boolean {\n    return this.enabled;\n  }\n\n  /**\n   * Clear all persisted activations. Used for testing or admin reset.\n   */\n  clearAll(): void {\n    this.state = this.createEmptyState();\n    if (this.enabled) {\n      this.persistAsync();\n    }\n  }\n\n  /**\n   * Fire-and-forget persistence with retry for transient disk failures.\n   * Retries up to PERSIST_MAX_RETRIES times with a short delay.\n   * Disk failure does not block activation operations.\n   */\n  private persistAsync(): void {\n    this.persistWithRetry(0).catch(error => {\n      logger.warn('[ActivationStore] Failed to persist activation state after retries', { error });\n\n      SecurityMonitor.logSecurityEvent({\n        type: 'ELEMENT_ACTIVATED',\n        severity: 'MEDIUM',\n        source: 'ActivationStore.persistAsync',\n        details: `Failed to persist activation state for session '${this.sessionId}' after ${PERSIST_MAX_RETRIES + 1} attempts — state continues in-memory only`,\n        additionalData: { error: String(error), sessionId: this.sessionId },\n      });\n    });\n  }\n\n  /**\n   * Attempt to persist with retries for transient failures (e.g., EBUSY, EAGAIN).\n   */\n  private async persistWithRetry(attempt: number): Promise<void> {\n    try {\n      await this.persist();\n    } catch (error) {\n      if (attempt < PERSIST_MAX_RETRIES) {\n        await new Promise(resolve => setTimeout(resolve, PERSIST_RETRY_DELAY_MS));\n        return this.persistWithRetry(attempt + 1);\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Write current activation state to disk.\n   */\n  private async persist(): Promise<void> {\n    this.state.lastUpdated = new Date().toISOString();\n    await fs.mkdir(this.stateDir, { recursive: true });\n    await this.fileOps.writeFile(this.persistPath, JSON.stringify(this.state, null, 2));\n  }\n\n  private createEmptyState(): PersistedActivationState {\n    return {\n      version: 1,\n      sessionId: this.sessionId,\n      lastUpdated: new Date().toISOString(),\n      activations: {},\n    };\n  }\n\n  private getTotalActivationCount(): number {\n    return Object.values(this.state.activations).reduce(\n      (sum, arr) => sum + (arr?.length ?? 0), 0\n    );\n  }\n\n  private getActivationCounts(): Record<string, number> {\n    const counts: Record<string, number> = {};\n    for (const [type, arr] of Object.entries(this.state.activations)) {\n      if (arr && arr.length > 0) {\n        counts[type] = arr.length;\n      }\n    }\n    return counts;\n  }\n}\n"]}
|
|
@@ -72,6 +72,7 @@ export interface SessionEventPayload {
|
|
|
72
72
|
export interface IngestBroadcasts {
|
|
73
73
|
logBroadcast: (entry: UnifiedLogEntry) => void;
|
|
74
74
|
metricsOnSnapshot?: (snapshot: MetricSnapshot) => void;
|
|
75
|
+
storeMetricsSnapshot?: (snapshot: MetricSnapshot, sessionId: string) => void;
|
|
75
76
|
sessionBroadcast?: (event: SessionInfo) => void;
|
|
76
77
|
}
|
|
77
78
|
/**
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"IngestRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAgB,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAE1C,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAC9D,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AA0B7D;;GAEG;AACH,MAAM,WAAW,WAAW;IAC1B,oEAAoE;IACpE,SAAS,EAAE,MAAM,CAAC;IAClB,uEAAuE;IACvE,WAAW,EAAE,MAAM,CAAC;IACpB,qDAAqD;IACrD,KAAK,EAAE,MAAM,CAAC;IACd,8DAA8D;IAC9D,GAAG,EAAE,MAAM,CAAC;IACZ,8CAA8C;IAC9C,SAAS,EAAE,MAAM,CAAC;IAClB,+FAA+F;IAC/F,aAAa,EAAE,MAAM,CAAC;IACtB,uEAAuE;IACvE,MAAM,EAAE,QAAQ,GAAG,OAAO,CAAC;IAC3B,wEAAwE;IACxE,QAAQ,EAAE,OAAO,CAAC;IAClB,wEAAwE;IACxE,aAAa,EAAE,OAAO,CAAC;IACvB,iGAAiG;IACjG,IAAI,EAAE,KAAK,GAAG,SAAS,CAAC;CACzB;AAED;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,eAAe,EAAE,CAAC;CAC5B;AAED;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,cAAc,CAAC;CAC1B;AAED;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,SAAS,GAAG,SAAS,GAAG,WAAW,CAAC;IAC3C,GAAG,EAAE,MAAM,CAAC;IACZ,SAAS,EAAE,MAAM,CAAC;CACnB;AAED;;;GAGG;AACH,MAAM,WAAW,gBAAgB;IAC/B,YAAY,EAAE,CAAC,KAAK,EAAE,eAAe,KAAK,IAAI,CAAC;IAC/C,iBAAiB,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,KAAK,IAAI,CAAC;IACvD,gBAAgB,CAAC,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,IAAI,CAAC;CACjD;AAED;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC,MAAM,EAAE,MAAM,CAAC;IACf,+BAA+B;IAC/B,WAAW,EAAE,MAAM,WAAW,EAAE,CAAC;IACjC,uCAAuC;IACvC,qBAAqB,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,KAAK,IAAI,CAAC;IAChE,oFAAoF;IACpF,sBAAsB,EAAE,MAAM,IAAI,CAAC;CACpC;AAOD;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAAC,UAAU,EAAE,gBAAgB,GAAG,kBAAkB,
|
|
1
|
+
{"version":3,"file":"IngestRoutes.d.ts","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAgB,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAE1C,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAC9D,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AA0B7D;;GAEG;AACH,MAAM,WAAW,WAAW;IAC1B,oEAAoE;IACpE,SAAS,EAAE,MAAM,CAAC;IAClB,uEAAuE;IACvE,WAAW,EAAE,MAAM,CAAC;IACpB,qDAAqD;IACrD,KAAK,EAAE,MAAM,CAAC;IACd,8DAA8D;IAC9D,GAAG,EAAE,MAAM,CAAC;IACZ,8CAA8C;IAC9C,SAAS,EAAE,MAAM,CAAC;IAClB,+FAA+F;IAC/F,aAAa,EAAE,MAAM,CAAC;IACtB,uEAAuE;IACvE,MAAM,EAAE,QAAQ,GAAG,OAAO,CAAC;IAC3B,wEAAwE;IACxE,QAAQ,EAAE,OAAO,CAAC;IAClB,wEAAwE;IACxE,aAAa,EAAE,OAAO,CAAC;IACvB,iGAAiG;IACjG,IAAI,EAAE,KAAK,GAAG,SAAS,CAAC;CACzB;AAED;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,eAAe,EAAE,CAAC;CAC5B;AAED;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,cAAc,CAAC;CAC1B;AAED;;GAEG;AACH,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,SAAS,GAAG,SAAS,GAAG,WAAW,CAAC;IAC3C,GAAG,EAAE,MAAM,CAAC;IACZ,SAAS,EAAE,MAAM,CAAC;CACnB;AAED;;;GAGG;AACH,MAAM,WAAW,gBAAgB;IAC/B,YAAY,EAAE,CAAC,KAAK,EAAE,eAAe,KAAK,IAAI,CAAC;IAC/C,iBAAiB,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,KAAK,IAAI,CAAC;IACvD,oBAAoB,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,KAAK,IAAI,CAAC;IAC7E,gBAAgB,CAAC,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,IAAI,CAAC;CACjD;AAED;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC,MAAM,EAAE,MAAM,CAAC;IACf,+BAA+B;IAC/B,WAAW,EAAE,MAAM,WAAW,EAAE,CAAC;IACjC,uCAAuC;IACvC,qBAAqB,EAAE,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,KAAK,IAAI,CAAC;IAChE,oFAAoF;IACpF,sBAAsB,EAAE,MAAM,IAAI,CAAC;CACpC;AAOD;;;;;GAKG;AACH,wBAAgB,kBAAkB,CAAC,UAAU,EAAE,gBAAgB,GAAG,kBAAkB,CAqanF"}
|
|
@@ -190,6 +190,9 @@ export function createIngestRoutes(broadcasts) {
|
|
|
190
190
|
if (broadcasts.metricsOnSnapshot) {
|
|
191
191
|
broadcasts.metricsOnSnapshot(payload.snapshot);
|
|
192
192
|
}
|
|
193
|
+
if (broadcasts.storeMetricsSnapshot) {
|
|
194
|
+
broadcasts.storeMetricsSnapshot(payload.snapshot, payload.sessionId);
|
|
195
|
+
}
|
|
193
196
|
// Update heartbeat, revive ended sessions, or auto-register orphans (#1870)
|
|
194
197
|
const session = ensureSession(payload.sessionId);
|
|
195
198
|
logger.debug(`[IngestRoutes] Metrics ingested from ${session?.displayName ?? payload.sessionId}`);
|
|
@@ -447,4 +450,4 @@ export function createIngestRoutes(broadcasts) {
|
|
|
447
450
|
}
|
|
448
451
|
return { router, getSessions, registerLeaderSession, registerConsoleSession };
|
|
449
452
|
}
|
|
450
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"IngestRoutes.js","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAO,OAAO,EAAE,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAI1C,OAAO,EAAE,wBAAwB,EAAE,MAAM,yCAAyC,CAAC;AACnF,OAAO,EAAE,gBAAgB,EAAE,MAAM,+CAA+C,CAAC;AACjF,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AACpD,OAAO,EAAE,MAAM,EAAE,MAAM,uBAAuB,CAAC;AAC/C,OAAO,EAAE,GAAG,EAAE,MAAM,qBAAqB,CAAC;AAE1C,kDAAkD;AAClD,MAAM,gBAAgB,GAAG,KAAK,CAAC;AAE/B,qDAAqD;AACrD,MAAM,cAAc,GAAG,IAAI,CAAC;AAC5B,MAAM,oBAAoB,GAAG,MAAM,CAAC;AAEpC,iDAAiD;AACjD,MAAM,kBAAkB,GAAG,KAAK,CAAC;AAEjC,6EAA6E;AAC7E,MAAM,gBAAgB,GAAG,MAAM,CAAC;AAEhC,6DAA6D;AAC7D,MAAM,uBAAuB,GAAG,KAAK,CAAC;AAEtC,kEAAkE;AAClE,MAAM,cAAc,GAAG,CAAC,GAAG,MAAM,CAAC,CAAC,YAAY;AA6E/C,6DAA6D;AAC7D,SAAS,cAAc,CAAC,CAAS;IAC/B,OAAO,gBAAgB,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,iBAAiB,CAAC;AACzD,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,kBAAkB,CAAC,UAA4B;IAC7D,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC;IACxB,MAAM,QAAQ,GAAG,IAAI,GAAG,EAAuB,CAAC;IAChD,MAAM,QAAQ,GAAG,IAAI,eAAe,EAAE,CAAC;IACvC,MAAM,WAAW,GAAG,IAAI,wBAAwB,CAAC,cAAc,EAAE,oBAAoB,CAAC,CAAC;IAEvF,iEAAiE;IACjE,mFAAmF;IACnF,MAAM,cAAc,GAAG,IAAI,GAAG,EAAU,CAAC;IAEzC,6DAA6D;IAC7D,6EAA6E;IAC7E,mFAAmF;IACnF,MAAM,YAAY,GAAG,IAAI,GAAG,EAAU,CAAC;IAEvC,oDAAoD;IACpD,SAAS,qBAAqB,CAAC,SAAiB,EAAE,GAAY;QAC5D,MAAM,OAAO,GAAG,GAAG,IAAI,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,GAAG,CAAC;QACpD,IAAI,CAAC,OAAO;YAAE,OAAO;QACrB,IAAI,CAAC;YAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,kBAAkB,CAAC,CAAC;QACtE,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QACzC,IAAI,QAAQ;YAAE,QAAQ,CAAC,MAAM,GAAG,OAAO,CAAC;QACxC,MAAM,CAAC,IAAI,CAAC,qDAAqD,EAAE;YACjE,WAAW,EAAE,QAAQ,EAAE,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO;SAC5D,CAAC,CAAC;IACL,CAAC;IAED,2CAA2C;IAC3C,SAAS,mBAAmB,CAAC,SAAiB,EAAE,GAAY;QAC1D,qBAAqB,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC;QACtC,YAAY,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAC/B,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IAChC,CAAC;IAED,yEAAyE;IACzE,SAAS,YAAY,CAAC,SAAiB,EAAE,GAAY,EAAE,aAAa,GAAG,KAAK;QAC1E,IAAI,CAAC;YACH,MAAM,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;YAC/C,MAAM,KAAK,GAAG,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC;YACxD,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;YACrC,MAAM,IAAI,GAAgB;gBACxB,SAAS,EAAE,WAAW,EAAE,KAAK;gBAC7B,GAAG,EAAE,GAAG,IAAI,CAAC;gBACb,SAAS,EAAE,GAAG,EAAE,aAAa,EAAE,GAAG;gBAClC,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,aAAa,EAAE,IAAI,EAAE,KAAK;aAC9D,CAAC;YACF,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;YAC9B,MAAM,CAAC,IAAI,CAAC,iDAAiD,EAAE;gBAC7D,WAAW,EAAE,SAAS,EAAE,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,WAAW;aAChE,CAAC,CAAC;YACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,IAAI,CAAC,CAAC;YACpC,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,KAAK,CAAC,yDAAyD,EAAE;gBACtE,SAAS,EAAE,KAAK,EAAG,GAAa,CAAC,OAAO;aACzC,CAAC,CAAC;YACH,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,SAAS,aAAa,CAAC,SAAiB,EAAE,GAAY,EAAE,aAAa,GAAG,KAAK;QAC3E,IAAI,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC;YAAE,OAAO,IAAI,CAAC;QAC/C,IAAI,YAAY,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC;YAChC,mBAAmB,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC;YACpC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QACzC,IAAI,CAAC,QAAQ;YAAE,OAAO,YAAY,CAAC,SAAS,EAAE,GAAG,EAAE,aAAa,CAAC,CAAC;QAElE,IAAI,QAAQ,CAAC,MAAM,KAAK,OAAO,EAAE,CAAC;YAChC,QAAQ,CAAC,MAAM,GAAG,QAAQ,CAAC;YAC3B,MAAM,CAAC,IAAI,CAAC,yDAAyD,EAAE;gBACrE,WAAW,EAAE,QAAQ,CAAC,WAAW,EAAE,SAAS;aAC7C,CAAC,CAAC;QACL,CAAC;QACD,QAAQ,CAAC,aAAa,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAClD,IAAI,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC;YACzB,QAAQ,CAAC,GAAG,GAAG,GAAG,CAAC;YACnB,MAAM,CAAC,IAAI,CAAC,mDAAmD,EAAE;gBAC/D,WAAW,EAAE,QAAQ,CAAC,WAAW,EAAE,SAAS,EAAE,GAAG;aAClD,CAAC,CAAC;QACL,CAAC;QACD,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,oCAAoC;IACpC,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,gBAAgB,EAAE,CAAC,CAAC,CAAC;IAEtD;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QAC9D,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,EAAE,CAAC;YAC9B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,qBAAqB,EAAE,CAAC,CAAC;YACvD,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,GAAG,CAAC,IAAwB,CAAC;QAC7C,IAAI,CAAC,OAAO,EAAE,SAAS,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YAC3D,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACrD,MAAM,CAAC,IAAI,CAAC,oCAAoC,EAAE,EAAE,QAAQ,EAAE,YAAY,EAAE,CAAC,CAAC,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,KAAK,CAAC,OAAO,CAAC,OAAO,EAAE,OAAO,CAAC,EAAE,CAAC,CAAC;YACjJ,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,SAAS,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;YACjG,OAAO;QACT,CAAC;QACD,OAAO,CAAC,SAAS,GAAG,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEtD,IAAI,KAAK,GAAG,CAAC,CAAC;QACd,IAAI,OAAO,GAAG,CAAC,CAAC;QAChB,KAAK,MAAM,KAAK,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;YACpC,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,CAAC,OAAO,KAAK,QAAQ,EAAE,CAAC;gBAAC,OAAO,EAAE,CAAC;gBAAC,SAAS;YAAC,CAAC;YACzE,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,UAAU,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;YACjC,KAAK,EAAE,CAAC;QACV,CAAC;QAED,4EAA4E;QAC5E,MAAM,OAAO,GAAG,aAAa,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEjD,IAAI,OAAO,GAAG,CAAC,EAAE,CAAC;YAChB,MAAM,CAAC,KAAK,CAAC,kCAAkC,OAAO,EAAE,WAAW,IAAI,OAAO,CAAC,SAAS,cAAc,KAAK,aAAa,OAAO,EAAE,CAAC,CAAC;QACrI,CAAC;QAED,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;IACrD,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QACjE,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,EAAE,CAAC;YAC9B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,qBAAqB,EAAE,CAAC,CAAC;YACvD,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,GAAG,CAAC,IAA4B,CAAC;QACjD,IAAI,CAAC,OAAO,EAAE,SAAS,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;YAC7C,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACrD,MAAM,CAAC,IAAI,CAAC,wCAAwC,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC;YACpE,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,UAAU,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;YAClG,OAAO;QACT,CAAC;QACD,OAAO,CAAC,SAAS,GAAG,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEtD,IAAI,UAAU,CAAC,iBAAiB,EAAE,CAAC;YACjC,UAAU,CAAC,iBAAiB,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QACjD,CAAC;QAED,4EAA4E;QAC5E,MAAM,OAAO,GAAG,aAAa,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QACjD,MAAM,CAAC,KAAK,CAAC,wCAAwC,OAAO,EAAE,WAAW,IAAI,OAAO,CAAC,SAAS,EAAE,CAAC,CAAC;QAClG,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;QACjE,MAAM,OAAO,GAAG,GAAG,CAAC,IAA2B,CAAC;QAChD,IAAI,CAAC,OAAO,EAAE,SAAS,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;YAC1C,MAAM,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACrD,MAAM,CAAC,IAAI,CAAC,8CAA8C,EAAE,EAAE,QAAQ,EAAE,CAAC,CAAC;YAC1E,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,OAAO,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;YAC/F,OAAO;QACT,CAAC;QACD,OAAO,CAAC,SAAS,GAAG,cAAc,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAEtD,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAErC,QAAQ,OAAO,CAAC,KAAK,EAAE,CAAC;YACtB,KAAK,SAAS,CAAC,CAAC,CAAC;gBACf,iEAAiE;gBACjE,IAAI,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC;oBAAE,MAAM;gBACjD,IAAI,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;oBAAC,mBAAmB,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;oBAAC,MAAM;gBAAC,CAAC;gBAExG,MAAM,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;gBACvD,MAAM,KAAK,GAAG,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC;gBAChE,MAAM,eAAe,GAAG,OAAO,CAAE,GAAW,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;gBACjE,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,EAAE;oBAC9B,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,WAAW,EAAE,KAAK;oBAChD,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,SAAS,EAAE,OAAO,CAAC,SAAS,IAAI,GAAG,EAAE,aAAa,EAAE,GAAG;oBACzE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,aAAa,EAAE,eAAe,EAAE,IAAI,EAAE,KAAK;iBAC/E,CAAC,CAAC;gBACH,MAAM,CAAC,IAAI,CAAC,mCAAmC,EAAE;oBAC/C,WAAW,EAAE,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,KAAK;oBAClE,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM;iBACxF,CAAC,CAAC;gBACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAE,CAAC,CAAC;gBAChE,MAAM;YACR,CAAC;YACD,KAAK,SAAS,CAAC,CAAC,CAAC;gBACf,MAAM,QAAQ,GAAG,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;gBACjD,IAAI,QAAQ,EAAE,CAAC;oBACb,QAAQ,CAAC,MAAM,GAAG,OAAO,CAAC;oBAC1B,QAAQ,CAAC,aAAa,GAAG,GAAG,CAAC;oBAC7B,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;oBACpC,MAAM,CAAC,IAAI,CAAC,gCAAgC,EAAE;wBAC5C,WAAW,EAAE,QAAQ,CAAC,WAAW,EAAE,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,QAAQ,CAAC,GAAG;wBAClF,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC;qBAC5F,CAAC,CAAC;oBACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC,CAAC;gBAC1C,CAAC;gBACD,MAAM;YACR,CAAC;YACD,KAAK,WAAW,CAAC,CAAC,CAAC;gBACjB,wEAAwE;gBACxE,aAAa,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC;gBAC9C,MAAM;YACR,CAAC;QACH,CAAC;QAED,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC;IACrC,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,GAAG,CAAC,eAAe,EAAE,KAAK,EAAE,IAAa,EAAE,GAAa,EAAE,EAAE;QACjE,4EAA4E;QAC5E,0DAA0D;QAC1D,MAAM,aAAa,GAAG,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC;QACvF,MAAM,WAAW,GAAG,GAAG,CAAC,0BAA0B,IAAI,KAAK,CAAC;QAE5D,mEAAmE;QACnE,kEAAkE;QAClE,qDAAqD;QACrD,IAAI,WAAW,KAAK,IAAI,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,MAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;gBACzC,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,uBAAuB,CAAC,CAAC;gBAC9E,MAAM,SAAS,GAAG,MAAM,KAAK,CAAC,oCAAoC,EAAE;oBAClE,MAAM,EAAE,UAAU,CAAC,MAAM;iBAC1B,CAAC,CAAC;gBACH,YAAY,CAAC,OAAO,CAAC,CAAC;gBACtB,IAAI,SAAS,CAAC,EAAE,EAAE,CAAC;oBACjB,MAAM,UAAU,GAAG,MAAM,SAAS,CAAC,IAAI,EAAiC,CAAC;oBACzE,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;oBAC9D,KAAK,MAAM,EAAE,IAAI,CAAC,UAAU,CAAC,QAAQ,IAAI,EAAE,CAAC,EAAE,CAAC;wBAC7C,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;4BAC1D,aAAa,CAAC,IAAI,CAAC;gCACjB,GAAG,EAAE;gCACL,aAAa,EAAE,KAAK;gCACpB,IAAI,EAAE,EAAE,CAAC,IAAI,IAAI,KAAK;6BACvB,CAAC,CAAC;wBACL,CAAC;oBACH,CAAC;gBACH,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,2DAA2D;YAC7D,CAAC;QACH,CAAC;QAED,GAAG,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,aAAa,EAAE,CAAC,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH;;OAEG;IACH,MAAM,CAAC,IAAI,CAAC,+BAA+B,EAAE,KAAK,EAAE,GAAY,EAAE,GAAa,EAAE,EAAE;QACjF,MAAM,SAAS,GAAG,GAAG,CAAC,MAAM,CAAC,WAAW,CAAW,CAAC;QACpD,MAAM,OAAO,GAAG,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAExC,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,sEAAsE;YACtE,MAAM,WAAW,GAAG,GAAG,CAAC,0BAA0B,IAAI,KAAK,CAAC;YAC5D,IAAI,WAAW,KAAK,IAAI,EAAE,CAAC;gBACzB,IAAI,CAAC;oBACH,MAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;oBACzC,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,uBAAuB,CAAC,CAAC;oBAC9E,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,sCAAsC,kBAAkB,CAAC,SAAS,CAAC,OAAO,EAAE;wBACvG,MAAM,EAAE,MAAM;wBACd,MAAM,EAAE,UAAU,CAAC,MAAM;qBAC1B,CAAC,CAAC;oBACH,YAAY,CAAC,OAAO,CAAC,CAAC;oBACtB,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;wBAChB,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;wBACnC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;wBACf,OAAO;oBACT,CAAC;gBACH,CAAC;gBAAC,MAAM,CAAC;oBACP,oDAAoD;gBACtD,CAAC;YACH,CAAC;YACD,MAAM,CAAC,IAAI,CAAC,mDAAmD,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;YAChF,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,SAAS,EAAE,CAAC,CAAC;YAChE,OAAO;QACT,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC;YACjB,6EAA6E;YAC7E,wEAAwE;YACxE,4EAA4E;YAC5E,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC;YACzB,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;YAC5B,YAAY,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YAC5B,MAAM,CAAC,IAAI,CAAC,qEAAqE,EAAE;gBACjF,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS;aAC5C,CAAC,CAAC;YACH,GAAG,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,CAAC,WAAW,EAAE,MAAM,EAAE,cAAc,EAAE,CAAC,CAAC;YAC/E,OAAO;QACT,CAAC;QAED,kFAAkF;QAClF,wEAAwE;QACxE,IAAI,MAAM,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC;YACH,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,SAAS,CAAC,CAAC;YACrC,MAAM,GAAG,IAAI,CAAC;QAChB,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,IAAI,GAAI,GAA6B,CAAC,IAAI,CAAC;YACjD,IAAI,IAAI,KAAK,OAAO,EAAE,CAAC;gBACrB,MAAM,GAAG,IAAI,CAAC,CAAC,kDAAkD;YACnE,CAAC;iBAAM,CAAC;gBACN,MAAM,CAAC,KAAK,CAAC,uCAAuC,EAAE;oBACpD,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,KAAK,EAAG,GAAa,CAAC,OAAO;iBAC7F,CAAC,CAAC;gBACH,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,wBAAwB,EAAE,SAAS,EAAE,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,MAAM,EAAG,GAAa,CAAC,OAAO,EAAE,CAAC,CAAC;gBACzJ,OAAO;YACT,CAAC;QACH,CAAC;QACD,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC;QACzB,QAAQ,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAC5B,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC9B,MAAM,CAAC,IAAI,CAAC,+BAA+B,EAAE;YAC3C,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG;YAC7D,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC;SAC5F,CAAC,CAAC;QACH,GAAG,CAAC,IAAI,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,WAAW,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IACxE,CAAC,CAAC,CAAC;IAEH,2CAA2C;IAC3C,SAAS,iBAAiB,CAAC,GAAW;QACpC,KAAK,MAAM,CAAC,EAAE,EAAE,OAAO,CAAC,IAAI,QAAQ,EAAE,CAAC;YACrC,IAAI,OAAO,CAAC,MAAM,KAAK,QAAQ;gBAAE,SAAS;YAC1C,IAAI,OAAO,CAAC,QAAQ,IAAI,OAAO,CAAC,IAAI,KAAK,SAAS;gBAAE,SAAS;YAC7D,MAAM,GAAG,GAAG,GAAG,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,OAAO,EAAE,CAAC;YAC5D,IAAI,GAAG,IAAI,gBAAgB;gBAAE,SAAS;YACtC,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC;YACzB,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;YACrB,MAAM,CAAC,IAAI,CAAC,qCAAqC,EAAE;gBACjD,WAAW,EAAE,OAAO,CAAC,WAAW,EAAE,SAAS,EAAE,EAAE,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG;gBACjE,gBAAgB,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,IAAI,CAAC,GAAG;gBAC9C,cAAc,EAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,GAAG,CAAC;aAC5F,CAAC,CAAC;YACH,UAAU,CAAC,gBAAgB,EAAE,CAAC,OAAO,CAAC,CAAC;QACzC,CAAC;IACH,CAAC;IAED,qDAAqD;IACrD,SAAS,iBAAiB,CAAC,GAAW;QACpC,KAAK,MAAM,CAAC,EAAE,EAAE,OAAO,CAAC,IAAI,QAAQ,EAAE,CAAC;YACrC,IAAI,OAAO,CAAC,MAAM,KAAK,OAAO,IAAI,GAAG,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,OAAO,EAAE,GAAG,cAAc,EAAE,CAAC;gBACnG,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YACtB,CAAC;QACH,CAAC;IACH,CAAC;IAED,MAAM,cAAc,GAAG,WAAW,CAAC,GAAG,EAAE;QACtC,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,iBAAiB,CAAC,GAAG,CAAC,CAAC;QACvB,iBAAiB,CAAC,GAAG,CAAC,CAAC;IACzB,CAAC,EAAE,kBAAkB,CAAC,CAAC;IACvB,cAAc,CAAC,KAAK,EAAE,CAAC;IAEvB,SAAS,WAAW;QAClB,OAAO,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC;IAC1E,CAAC;IAED,SAAS,qBAAqB,CAAC,SAAiB,EAAE,GAAW;QAC3D,MAAM,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;QACrD,MAAM,KAAK,GAAG,QAAQ,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,SAAS,CAAC;QACxD,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE;YACtB,SAAS;YACT,WAAW;YACX,KAAK;YACL,GAAG;YACH,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACnC,aAAa,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACvC,MAAM,EAAE,QAAQ;YAChB,QAAQ,EAAE,IAAI;YACd,aAAa,EAAE,IAAI;YACnB,IAAI,EAAE,KAAK;SACZ,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,0CAA0C,EAAE,EAAE,WAAW,EAAE,SAAS,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC;IAClG,CAAC;IAED;;;;OAIG;IACH,SAAS,sBAAsB;QAC7B,MAAM,SAAS,GAAG,WAAW,OAAO,CAAC,GAAG,EAAE,CAAC;QAC3C,IAAI,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC;YAAE,OAAO;QACpC,MAAM,WAAW,GAAG,aAAa,CAAC;QAClC,QAAQ,CAAC,GAAG,CAAC,SAAS,EAAE;YACtB,SAAS,EAAE,SAAS;YACpB,WAAW;YACX,KAAK,EAAE,SAAS,EAAE,6CAA6C;YAC/D,GAAG,EAAE,OAAO,CAAC,GAAG;YAChB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACnC,aAAa,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACvC,MAAM,EAAE,QAAQ;YAChB,QAAQ,EAAE,KAAK;YACf,aAAa,EAAE,IAAI;YACnB,IAAI,EAAE,SAAS;SAChB,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,2CAA2C,EAAE,EAAE,SAAS,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;IACvG,CAAC;IAED,OAAO,EAAE,MAAM,EAAE,WAAW,EAAE,qBAAqB,EAAE,sBAAsB,EAAE,CAAC;AAChF,CAAC","sourcesContent":["/**\n * Event ingestion routes for the unified web console.\n *\n * The console leader mounts these routes so follower MCP servers can\n * forward their logs, metrics, and session lifecycle events. All ingested\n * entries are stamped with `_sessionId` in their data field and then\n * broadcast to SSE clients via the existing log/metrics broadcast hooks.\n *\n * Routes:\n * - POST /api/ingest/logs     — Batched log entries from a follower\n * - POST /api/ingest/metrics  — Metric snapshots from a follower\n * - POST /api/ingest/session  — Session lifecycle events (started/stopped/heartbeat)\n * - GET  /api/sessions        — Active session list for the UI\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport express, { Router } from 'express';\nimport type { Request, Response } from 'express';\nimport type { UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { SessionNamePool } from './SessionNames.js';\nimport { logger } from '../../utils/logger.js';\nimport { env } from '../../config/env.js';\n\n/** Maximum payload size for ingestion requests */\nconst MAX_PAYLOAD_SIZE = '1mb';\n\n/** Rate limit: max requests per window per source */\nconst RATE_LIMIT_MAX = 1000;\nconst RATE_LIMIT_WINDOW_MS = 60_000;\n\n/** How often to check for stale sessions (ms) */\nconst REAPER_INTERVAL_MS = 5_000;\n\n/** How long since last heartbeat before a session is considered dead (ms) */\nconst SESSION_STALE_MS = 15_000;\n\n/** Timeout for legacy port federation/proxy requests (ms) */\nconst LEGACY_FETCH_TIMEOUT_MS = 2_000;\n\n/** How long before ended sessions are purged from the Map (ms) */\nconst ENDED_PURGE_MS = 5 * 60_000; // 5 minutes\n\n/**\n * Tracked session information.\n */\nexport interface SessionInfo {\n  /** Unique identifier for this session (UUID or `console-<pid>`). */\n  sessionId: string;\n  /** Friendly puppet name (e.g., \"Kermit\", \"Punch\") or \"Web Console\". */\n  displayName: string;\n  /** Canonical hex color for this puppet character. */\n  color: string;\n  /** OS process ID of the MCP server or web console process. */\n  pid: number;\n  /** ISO timestamp when the session started. */\n  startedAt: string;\n  /** ISO timestamp of the most recent heartbeat (followers) or registration (leader/console). */\n  lastHeartbeat: string;\n  /** Lifecycle status — 'active' until ended or reaped for staleness. */\n  status: 'active' | 'ended';\n  /** True if this session won leader election and owns the token file. */\n  isLeader: boolean;\n  /** Whether this session connected with a valid Bearer token (#1805). */\n  authenticated: boolean;\n  /** Session kind — 'mcp' for MCP stdio sessions, 'console' for the web console itself (#1805). */\n  kind: 'mcp' | 'console';\n}\n\n/**\n * Payload for POST /api/ingest/logs\n */\nexport interface IngestLogPayload {\n  sessionId: string;\n  entries: UnifiedLogEntry[];\n}\n\n/**\n * Payload for POST /api/ingest/metrics\n */\nexport interface IngestMetricsPayload {\n  sessionId: string;\n  snapshot: MetricSnapshot;\n}\n\n/**\n * Payload for POST /api/ingest/session\n */\nexport interface SessionEventPayload {\n  sessionId: string;\n  event: 'started' | 'stopped' | 'heartbeat';\n  pid: number;\n  startedAt: string;\n}\n\n/**\n * Callbacks provided by the unified console orchestrator for broadcasting\n * ingested events through the existing SSE infrastructure.\n */\nexport interface IngestBroadcasts {\n  logBroadcast: (entry: UnifiedLogEntry) => void;\n  metricsOnSnapshot?: (snapshot: MetricSnapshot) => void;\n  sessionBroadcast?: (event: SessionInfo) => void;\n}\n\n/**\n * Result of creating ingest routes.\n */\nexport interface IngestRoutesResult {\n  router: Router;\n  /** Get all tracked sessions */\n  getSessions: () => SessionInfo[];\n  /** Register the leader as a session */\n  registerLeaderSession: (sessionId: string, pid: number) => void;\n  /** Register the web console as a session so the indicator is never empty (#1805) */\n  registerConsoleSession: () => void;\n}\n\n/** Normalize a string via UnicodeValidator (DMCP-SEC-004) */\nfunction normalizeInput(s: string): string {\n  return UnicodeValidator.normalize(s).normalizedContent;\n}\n\n/**\n * Create the ingestion routes and session registry.\n *\n * @param broadcasts - Callbacks to forward ingested events to SSE clients\n * @returns Router and session management functions\n */\nexport function createIngestRoutes(broadcasts: IngestBroadcasts): IngestRoutesResult {\n  const router = Router();\n  const sessions = new Map<string, SessionInfo>();\n  const namePool = new SessionNamePool();\n  const rateLimiter = new SlidingWindowRateLimiter(RATE_LIMIT_MAX, RATE_LIMIT_WINDOW_MS);\n\n  // Sessions the user explicitly killed — never come back (#1870).\n  // Cleared only on server restart, which is appropriate since that's a new context.\n  const killedSessions = new Set<string>();\n\n  // Sessions waiting for a PID so we can SIGTERM them (#1870).\n  // When the user dismisses a pid=0 orphan, we add it here. The next heartbeat\n  // (every 10s) carries the PID — we SIGTERM immediately and move to killedSessions.\n  const pendingKills = new Set<string>();\n\n  /** Execute a deferred kill if we now have a PID. */\n  function tryExecutePendingKill(sessionId: string, pid?: number): void {\n    const killPid = pid || sessions.get(sessionId)?.pid;\n    if (!killPid) return;\n    try { process.kill(killPid, 'SIGTERM'); } catch { /* already dead */ }\n    const existing = sessions.get(sessionId);\n    if (existing) existing.status = 'ended';\n    logger.info('[IngestRoutes] Deferred kill executed — PID arrived', {\n      displayName: existing?.displayName, sessionId, pid: killPid,\n    });\n  }\n\n  /** Promote a pending kill to permanent. */\n  function finalizePendingKill(sessionId: string, pid?: number): void {\n    tryExecutePendingKill(sessionId, pid);\n    pendingKills.delete(sessionId);\n    killedSessions.add(sessionId);\n  }\n\n  /** Create a new session entry for an orphan. Returns null on failure. */\n  function autoRegister(sessionId: string, pid?: number, authenticated = false): SessionInfo | null {\n    try {\n      const displayName = namePool.assign(sessionId);\n      const color = namePool.getColor(sessionId) ?? '#3b82f6';\n      const now = new Date().toISOString();\n      const info: SessionInfo = {\n        sessionId, displayName, color,\n        pid: pid || 0,\n        startedAt: now, lastHeartbeat: now,\n        status: 'active', isLeader: false, authenticated, kind: 'mcp',\n      };\n      sessions.set(sessionId, info);\n      logger.info('[IngestRoutes] Auto-registered orphaned session', {\n        displayName, sessionId, source: pid ? 'heartbeat' : 'ingestion',\n      });\n      broadcasts.sessionBroadcast?.(info);\n      return info;\n    } catch (err) {\n      logger.debug('[IngestRoutes] Failed to auto-register orphaned session', {\n        sessionId, error: (err as Error).message,\n      });\n      return null;\n    }\n  }\n\n  /**\n   * Auto-register or update an orphaned session from ingestion data.\n   * Returns the session (existing or newly created), or null if killed/pending.\n   */\n  function ensureSession(sessionId: string, pid?: number, authenticated = false): SessionInfo | null {\n    if (killedSessions.has(sessionId)) return null;\n    if (pendingKills.has(sessionId)) {\n      finalizePendingKill(sessionId, pid);\n      return null;\n    }\n\n    const existing = sessions.get(sessionId);\n    if (!existing) return autoRegister(sessionId, pid, authenticated);\n\n    if (existing.status === 'ended') {\n      existing.status = 'active';\n      logger.info('[IngestRoutes] Revived ended session still sending data', {\n        displayName: existing.displayName, sessionId,\n      });\n    }\n    existing.lastHeartbeat = new Date().toISOString();\n    if (pid && !existing.pid) {\n      existing.pid = pid;\n      logger.info('[IngestRoutes] Recovered PID for orphaned session', {\n        displayName: existing.displayName, sessionId, pid,\n      });\n    }\n    return existing;\n  }\n\n  // JSON body parsing with size limit\n  router.use(express.json({ limit: MAX_PAYLOAD_SIZE }));\n\n  /**\n   * POST /api/ingest/logs — Receive batched log entries from a follower.\n   */\n  router.post('/api/ingest/logs', (req: Request, res: Response) => {\n    if (!rateLimiter.tryAcquire()) {\n      res.status(429).json({ error: 'Rate limit exceeded' });\n      return;\n    }\n\n    const payload = req.body as IngestLogPayload;\n    if (!payload?.sessionId || !Array.isArray(payload.entries)) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid log payload', { received, hasSessionId: !!payload?.sessionId, hasEntries: Array.isArray(payload?.entries) });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'entries'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    let count = 0;\n    let skipped = 0;\n    for (const entry of payload.entries) {\n      if (!entry || typeof entry.message !== 'string') { skipped++; continue; }\n      const stamped: UnifiedLogEntry = {\n        ...entry,\n        data: { ...entry.data, _sessionId: payload.sessionId },\n      };\n      broadcasts.logBroadcast(stamped);\n      count++;\n    }\n\n    // Update heartbeat, revive ended sessions, or auto-register orphans (#1870)\n    const session = ensureSession(payload.sessionId);\n\n    if (skipped > 0) {\n      logger.debug(`[IngestRoutes] Log ingest from ${session?.displayName ?? payload.sessionId}: accepted=${count}, skipped=${skipped}`);\n    }\n\n    res.status(200).json({ accepted: count, skipped });\n  });\n\n  /**\n   * POST /api/ingest/metrics — Receive metric snapshots from a follower.\n   */\n  router.post('/api/ingest/metrics', (req: Request, res: Response) => {\n    if (!rateLimiter.tryAcquire()) {\n      res.status(429).json({ error: 'Rate limit exceeded' });\n      return;\n    }\n\n    const payload = req.body as IngestMetricsPayload;\n    if (!payload?.sessionId || !payload.snapshot) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid metrics payload', { received });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'snapshot'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    if (broadcasts.metricsOnSnapshot) {\n      broadcasts.metricsOnSnapshot(payload.snapshot);\n    }\n\n    // Update heartbeat, revive ended sessions, or auto-register orphans (#1870)\n    const session = ensureSession(payload.sessionId);\n    logger.debug(`[IngestRoutes] Metrics ingested from ${session?.displayName ?? payload.sessionId}`);\n    res.status(200).json({ accepted: true });\n  });\n\n  /**\n   * POST /api/ingest/session — Session lifecycle events.\n   */\n  router.post('/api/ingest/session', (req: Request, res: Response) => {\n    const payload = req.body as SessionEventPayload;\n    if (!payload?.sessionId || !payload.event) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid session event payload', { received });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'event'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    const now = new Date().toISOString();\n\n    switch (payload.event) {\n      case 'started': {\n        // Killed sessions stay dead; pending kills get finalized (#1870)\n        if (killedSessions.has(payload.sessionId)) break;\n        if (pendingKills.has(payload.sessionId)) { finalizePendingKill(payload.sessionId, payload.pid); break; }\n\n        const displayName = namePool.assign(payload.sessionId);\n        const color = namePool.getColor(payload.sessionId) ?? '#3b82f6';\n        const isAuthenticated = Boolean((res as any).locals?.tokenEntry);\n        sessions.set(payload.sessionId, {\n          sessionId: payload.sessionId, displayName, color,\n          pid: payload.pid, startedAt: payload.startedAt || now, lastHeartbeat: now,\n          status: 'active', isLeader: false, authenticated: isAuthenticated, kind: 'mcp',\n        });\n        logger.info('[IngestRoutes] Session registered', {\n          displayName, sessionId: payload.sessionId, pid: payload.pid, color,\n          activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length,\n        });\n        broadcasts.sessionBroadcast?.(sessions.get(payload.sessionId)!);\n        break;\n      }\n      case 'stopped': {\n        const existing = sessions.get(payload.sessionId);\n        if (existing) {\n          existing.status = 'ended';\n          existing.lastHeartbeat = now;\n          namePool.release(payload.sessionId);\n          logger.info('[IngestRoutes] Session stopped', {\n            displayName: existing.displayName, sessionId: payload.sessionId, pid: existing.pid,\n            activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n          });\n          broadcasts.sessionBroadcast?.(existing);\n        }\n        break;\n      }\n      case 'heartbeat': {\n        // Auto-register or update — heartbeat includes PID for recovery (#1870)\n        ensureSession(payload.sessionId, payload.pid);\n        break;\n      }\n    }\n\n    res.status(200).json({ ok: true });\n  });\n\n  /**\n   * GET /api/sessions — List all tracked sessions.\n   */\n  router.get('/api/sessions', async (_req: Request, res: Response) => {\n    // Server-side active filter — the frontend also filters, but ended sessions\n    // should never leave the API to prevent stale UI (#1870).\n    const localSessions = Array.from(sessions.values()).filter(s => s.status === 'active');\n    const currentPort = env.DOLLHOUSE_WEB_CONSOLE_PORT ?? 41715;\n\n    // Federate with the legacy port (3939) to show all sessions on the\n    // machine, including unauthenticated ones from pre-auth installs.\n    // Server-to-server avoids CORS restrictions (#1805).\n    if (currentPort !== 3939) {\n      try {\n        const controller = new AbortController();\n        const timeout = setTimeout(() => controller.abort(), LEGACY_FETCH_TIMEOUT_MS);\n        const legacyRes = await fetch('http://127.0.0.1:3939/api/sessions', {\n          signal: controller.signal,\n        });\n        clearTimeout(timeout);\n        if (legacyRes.ok) {\n          const legacyData = await legacyRes.json() as { sessions: SessionInfo[] };\n          const localIds = new Set(localSessions.map(s => s.sessionId));\n          for (const ls of (legacyData.sessions || [])) {\n            if (!localIds.has(ls.sessionId) && ls.status === 'active') {\n              localSessions.push({\n                ...ls,\n                authenticated: false,\n                kind: ls.kind || 'mcp',\n              });\n            }\n          }\n        }\n      } catch {\n        // Legacy instance not running or unreachable — that's fine\n      }\n    }\n\n    res.json({ sessions: localSessions });\n  });\n\n  /**\n   * POST /api/sessions/:sessionId/kill — Terminate a session's server process.\n   */\n  router.post('/api/sessions/:sessionId/kill', async (req: Request, res: Response) => {\n    const sessionId = req.params['sessionId'] as string;\n    const session = sessions.get(sessionId);\n\n    if (!session) {\n      // Session not in local Map — try proxying kill to legacy port (#1870)\n      const currentPort = env.DOLLHOUSE_WEB_CONSOLE_PORT ?? 41715;\n      if (currentPort !== 3939) {\n        try {\n          const controller = new AbortController();\n          const timeout = setTimeout(() => controller.abort(), LEGACY_FETCH_TIMEOUT_MS);\n          const proxyRes = await fetch(`http://127.0.0.1:3939/api/sessions/${encodeURIComponent(sessionId)}/kill`, {\n            method: 'POST',\n            signal: controller.signal,\n          });\n          clearTimeout(timeout);\n          if (proxyRes.ok) {\n            const data = await proxyRes.json();\n            res.json(data);\n            return;\n          }\n        } catch {\n          // Legacy instance not running — fall through to 404\n        }\n      }\n      logger.warn('[IngestRoutes] Kill requested for unknown session', { sessionId });\n      res.status(404).json({ error: 'Session not found', sessionId });\n      return;\n    }\n\n    if (!session.pid) {\n      // Auto-registered orphan with unknown PID — queue for deferred kill (#1870).\n      // The next heartbeat (every ~10s) carries the PID. ensureSession() will\n      // SIGTERM the process as soon as the PID arrives. Session is gone for good.\n      session.status = 'ended';\n      namePool.release(sessionId);\n      pendingKills.add(sessionId);\n      logger.info('[IngestRoutes] Queued deferred kill — waiting for PID via heartbeat', {\n        displayName: session.displayName, sessionId,\n      });\n      res.json({ ok: true, dismissed: session.displayName, reason: 'pending-kill' });\n      return;\n    }\n\n    // SIGTERM the process. Even if it fails (ESRCH = already dead, EPERM = not ours),\n    // mark the session as permanently killed so it never reappears (#1870).\n    let killed = false;\n    try {\n      process.kill(session.pid, 'SIGTERM');\n      killed = true;\n    } catch (err) {\n      const code = (err as NodeJS.ErrnoException).code;\n      if (code === 'ESRCH') {\n        killed = true; // process already dead — treat as successful kill\n      } else {\n        logger.error('[IngestRoutes] Failed to kill session', {\n          displayName: session.displayName, sessionId, pid: session.pid, error: (err as Error).message,\n        });\n        res.status(500).json({ error: 'Failed to kill session', sessionId, displayName: session.displayName, pid: session.pid, detail: (err as Error).message });\n        return;\n      }\n    }\n    session.status = 'ended';\n    namePool.release(sessionId);\n    killedSessions.add(sessionId);\n    logger.info('[IngestRoutes] Session killed', {\n      displayName: session.displayName, sessionId, pid: session.pid,\n      activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n    });\n    res.json({ ok: true, killed: session.displayName, pid: session.pid });\n  });\n\n  /** Mark stale active sessions as ended. */\n  function reapStaleSessions(now: number): void {\n    for (const [id, session] of sessions) {\n      if (session.status !== 'active') continue;\n      if (session.isLeader || session.kind === 'console') continue;\n      const age = now - new Date(session.lastHeartbeat).getTime();\n      if (age <= SESSION_STALE_MS) continue;\n      session.status = 'ended';\n      namePool.release(id);\n      logger.info('[IngestRoutes] Reaped stale session', {\n        displayName: session.displayName, sessionId: id, pid: session.pid,\n        lastHeartbeatAgo: `${Math.round(age / 1000)}s`,\n        activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n      });\n      broadcasts.sessionBroadcast?.(session);\n    }\n  }\n\n  /** Delete ended sessions to bound memory (#1870). */\n  function purgeStaleEntries(now: number): void {\n    for (const [id, session] of sessions) {\n      if (session.status === 'ended' && now - new Date(session.lastHeartbeat).getTime() > ENDED_PURGE_MS) {\n        sessions.delete(id);\n      }\n    }\n  }\n\n  const reaperInterval = setInterval(() => {\n    const now = Date.now();\n    reapStaleSessions(now);\n    purgeStaleEntries(now);\n  }, REAPER_INTERVAL_MS);\n  reaperInterval.unref();\n\n  function getSessions(): SessionInfo[] {\n    return Array.from(sessions.values()).filter(s => s.status === 'active');\n  }\n\n  function registerLeaderSession(sessionId: string, pid: number): void {\n    const displayName = namePool.assign(sessionId, true);\n    const color = namePool.getColor(sessionId) ?? '#3b82f6';\n    sessions.set(sessionId, {\n      sessionId,\n      displayName,\n      color,\n      pid,\n      startedAt: new Date().toISOString(),\n      lastHeartbeat: new Date().toISOString(),\n      status: 'active',\n      isLeader: true,\n      authenticated: true,\n      kind: 'mcp',\n    });\n    logger.info('[IngestRoutes] Leader session registered', { displayName, sessionId, pid, color });\n  }\n\n  /**\n   * Register the web console itself as a session (#1805). Ensures the\n   * session indicator always shows at least one entry — the console the\n   * user is currently looking at.\n   */\n  function registerConsoleSession(): void {\n    const consoleId = `console-${process.pid}`;\n    if (sessions.has(consoleId)) return;\n    const displayName = 'Web Console';\n    sessions.set(consoleId, {\n      sessionId: consoleId,\n      displayName,\n      color: '#6366f1', // indigo — distinct from puppet greens/blues\n      pid: process.pid,\n      startedAt: new Date().toISOString(),\n      lastHeartbeat: new Date().toISOString(),\n      status: 'active',\n      isLeader: false,\n      authenticated: true,\n      kind: 'console',\n    });\n    logger.info('[IngestRoutes] Console session registered', { sessionId: consoleId, pid: process.pid });\n  }\n\n  return { router, getSessions, registerLeaderSession, registerConsoleSession };\n}\n"]}
|
|
453
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"IngestRoutes.js","sourceRoot":"","sources":["../../../src/web/console/IngestRoutes.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Event ingestion routes for the unified web console.\n *\n * The console leader mounts these routes so follower MCP servers can\n * forward their logs, metrics, and session lifecycle events. All ingested\n * entries are stamped with `_sessionId` in their data field and then\n * broadcast to SSE clients via the existing log/metrics broadcast hooks.\n *\n * Routes:\n * - POST /api/ingest/logs     — Batched log entries from a follower\n * - POST /api/ingest/metrics  — Metric snapshots from a follower\n * - POST /api/ingest/session  — Session lifecycle events (started/stopped/heartbeat)\n * - GET  /api/sessions        — Active session list for the UI\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport express, { Router } from 'express';\nimport type { Request, Response } from 'express';\nimport type { UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport { SlidingWindowRateLimiter } from '../../utils/SlidingWindowRateLimiter.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { SessionNamePool } from './SessionNames.js';\nimport { logger } from '../../utils/logger.js';\nimport { env } from '../../config/env.js';\n\n/** Maximum payload size for ingestion requests */\nconst MAX_PAYLOAD_SIZE = '1mb';\n\n/** Rate limit: max requests per window per source */\nconst RATE_LIMIT_MAX = 1000;\nconst RATE_LIMIT_WINDOW_MS = 60_000;\n\n/** How often to check for stale sessions (ms) */\nconst REAPER_INTERVAL_MS = 5_000;\n\n/** How long since last heartbeat before a session is considered dead (ms) */\nconst SESSION_STALE_MS = 15_000;\n\n/** Timeout for legacy port federation/proxy requests (ms) */\nconst LEGACY_FETCH_TIMEOUT_MS = 2_000;\n\n/** How long before ended sessions are purged from the Map (ms) */\nconst ENDED_PURGE_MS = 5 * 60_000; // 5 minutes\n\n/**\n * Tracked session information.\n */\nexport interface SessionInfo {\n  /** Unique identifier for this session (UUID or `console-<pid>`). */\n  sessionId: string;\n  /** Friendly puppet name (e.g., \"Kermit\", \"Punch\") or \"Web Console\". */\n  displayName: string;\n  /** Canonical hex color for this puppet character. */\n  color: string;\n  /** OS process ID of the MCP server or web console process. */\n  pid: number;\n  /** ISO timestamp when the session started. */\n  startedAt: string;\n  /** ISO timestamp of the most recent heartbeat (followers) or registration (leader/console). */\n  lastHeartbeat: string;\n  /** Lifecycle status — 'active' until ended or reaped for staleness. */\n  status: 'active' | 'ended';\n  /** True if this session won leader election and owns the token file. */\n  isLeader: boolean;\n  /** Whether this session connected with a valid Bearer token (#1805). */\n  authenticated: boolean;\n  /** Session kind — 'mcp' for MCP stdio sessions, 'console' for the web console itself (#1805). */\n  kind: 'mcp' | 'console';\n}\n\n/**\n * Payload for POST /api/ingest/logs\n */\nexport interface IngestLogPayload {\n  sessionId: string;\n  entries: UnifiedLogEntry[];\n}\n\n/**\n * Payload for POST /api/ingest/metrics\n */\nexport interface IngestMetricsPayload {\n  sessionId: string;\n  snapshot: MetricSnapshot;\n}\n\n/**\n * Payload for POST /api/ingest/session\n */\nexport interface SessionEventPayload {\n  sessionId: string;\n  event: 'started' | 'stopped' | 'heartbeat';\n  pid: number;\n  startedAt: string;\n}\n\n/**\n * Callbacks provided by the unified console orchestrator for broadcasting\n * ingested events through the existing SSE infrastructure.\n */\nexport interface IngestBroadcasts {\n  logBroadcast: (entry: UnifiedLogEntry) => void;\n  metricsOnSnapshot?: (snapshot: MetricSnapshot) => void;\n  storeMetricsSnapshot?: (snapshot: MetricSnapshot, sessionId: string) => void;\n  sessionBroadcast?: (event: SessionInfo) => void;\n}\n\n/**\n * Result of creating ingest routes.\n */\nexport interface IngestRoutesResult {\n  router: Router;\n  /** Get all tracked sessions */\n  getSessions: () => SessionInfo[];\n  /** Register the leader as a session */\n  registerLeaderSession: (sessionId: string, pid: number) => void;\n  /** Register the web console as a session so the indicator is never empty (#1805) */\n  registerConsoleSession: () => void;\n}\n\n/** Normalize a string via UnicodeValidator (DMCP-SEC-004) */\nfunction normalizeInput(s: string): string {\n  return UnicodeValidator.normalize(s).normalizedContent;\n}\n\n/**\n * Create the ingestion routes and session registry.\n *\n * @param broadcasts - Callbacks to forward ingested events to SSE clients\n * @returns Router and session management functions\n */\nexport function createIngestRoutes(broadcasts: IngestBroadcasts): IngestRoutesResult {\n  const router = Router();\n  const sessions = new Map<string, SessionInfo>();\n  const namePool = new SessionNamePool();\n  const rateLimiter = new SlidingWindowRateLimiter(RATE_LIMIT_MAX, RATE_LIMIT_WINDOW_MS);\n\n  // Sessions the user explicitly killed — never come back (#1870).\n  // Cleared only on server restart, which is appropriate since that's a new context.\n  const killedSessions = new Set<string>();\n\n  // Sessions waiting for a PID so we can SIGTERM them (#1870).\n  // When the user dismisses a pid=0 orphan, we add it here. The next heartbeat\n  // (every 10s) carries the PID — we SIGTERM immediately and move to killedSessions.\n  const pendingKills = new Set<string>();\n\n  /** Execute a deferred kill if we now have a PID. */\n  function tryExecutePendingKill(sessionId: string, pid?: number): void {\n    const killPid = pid || sessions.get(sessionId)?.pid;\n    if (!killPid) return;\n    try { process.kill(killPid, 'SIGTERM'); } catch { /* already dead */ }\n    const existing = sessions.get(sessionId);\n    if (existing) existing.status = 'ended';\n    logger.info('[IngestRoutes] Deferred kill executed — PID arrived', {\n      displayName: existing?.displayName, sessionId, pid: killPid,\n    });\n  }\n\n  /** Promote a pending kill to permanent. */\n  function finalizePendingKill(sessionId: string, pid?: number): void {\n    tryExecutePendingKill(sessionId, pid);\n    pendingKills.delete(sessionId);\n    killedSessions.add(sessionId);\n  }\n\n  /** Create a new session entry for an orphan. Returns null on failure. */\n  function autoRegister(sessionId: string, pid?: number, authenticated = false): SessionInfo | null {\n    try {\n      const displayName = namePool.assign(sessionId);\n      const color = namePool.getColor(sessionId) ?? '#3b82f6';\n      const now = new Date().toISOString();\n      const info: SessionInfo = {\n        sessionId, displayName, color,\n        pid: pid || 0,\n        startedAt: now, lastHeartbeat: now,\n        status: 'active', isLeader: false, authenticated, kind: 'mcp',\n      };\n      sessions.set(sessionId, info);\n      logger.info('[IngestRoutes] Auto-registered orphaned session', {\n        displayName, sessionId, source: pid ? 'heartbeat' : 'ingestion',\n      });\n      broadcasts.sessionBroadcast?.(info);\n      return info;\n    } catch (err) {\n      logger.debug('[IngestRoutes] Failed to auto-register orphaned session', {\n        sessionId, error: (err as Error).message,\n      });\n      return null;\n    }\n  }\n\n  /**\n   * Auto-register or update an orphaned session from ingestion data.\n   * Returns the session (existing or newly created), or null if killed/pending.\n   */\n  function ensureSession(sessionId: string, pid?: number, authenticated = false): SessionInfo | null {\n    if (killedSessions.has(sessionId)) return null;\n    if (pendingKills.has(sessionId)) {\n      finalizePendingKill(sessionId, pid);\n      return null;\n    }\n\n    const existing = sessions.get(sessionId);\n    if (!existing) return autoRegister(sessionId, pid, authenticated);\n\n    if (existing.status === 'ended') {\n      existing.status = 'active';\n      logger.info('[IngestRoutes] Revived ended session still sending data', {\n        displayName: existing.displayName, sessionId,\n      });\n    }\n    existing.lastHeartbeat = new Date().toISOString();\n    if (pid && !existing.pid) {\n      existing.pid = pid;\n      logger.info('[IngestRoutes] Recovered PID for orphaned session', {\n        displayName: existing.displayName, sessionId, pid,\n      });\n    }\n    return existing;\n  }\n\n  // JSON body parsing with size limit\n  router.use(express.json({ limit: MAX_PAYLOAD_SIZE }));\n\n  /**\n   * POST /api/ingest/logs — Receive batched log entries from a follower.\n   */\n  router.post('/api/ingest/logs', (req: Request, res: Response) => {\n    if (!rateLimiter.tryAcquire()) {\n      res.status(429).json({ error: 'Rate limit exceeded' });\n      return;\n    }\n\n    const payload = req.body as IngestLogPayload;\n    if (!payload?.sessionId || !Array.isArray(payload.entries)) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid log payload', { received, hasSessionId: !!payload?.sessionId, hasEntries: Array.isArray(payload?.entries) });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'entries'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    let count = 0;\n    let skipped = 0;\n    for (const entry of payload.entries) {\n      if (!entry || typeof entry.message !== 'string') { skipped++; continue; }\n      const stamped: UnifiedLogEntry = {\n        ...entry,\n        data: { ...entry.data, _sessionId: payload.sessionId },\n      };\n      broadcasts.logBroadcast(stamped);\n      count++;\n    }\n\n    // Update heartbeat, revive ended sessions, or auto-register orphans (#1870)\n    const session = ensureSession(payload.sessionId);\n\n    if (skipped > 0) {\n      logger.debug(`[IngestRoutes] Log ingest from ${session?.displayName ?? payload.sessionId}: accepted=${count}, skipped=${skipped}`);\n    }\n\n    res.status(200).json({ accepted: count, skipped });\n  });\n\n  /**\n   * POST /api/ingest/metrics — Receive metric snapshots from a follower.\n   */\n  router.post('/api/ingest/metrics', (req: Request, res: Response) => {\n    if (!rateLimiter.tryAcquire()) {\n      res.status(429).json({ error: 'Rate limit exceeded' });\n      return;\n    }\n\n    const payload = req.body as IngestMetricsPayload;\n    if (!payload?.sessionId || !payload.snapshot) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid metrics payload', { received });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'snapshot'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    if (broadcasts.metricsOnSnapshot) {\n      broadcasts.metricsOnSnapshot(payload.snapshot);\n    }\n    if (broadcasts.storeMetricsSnapshot) {\n      broadcasts.storeMetricsSnapshot(payload.snapshot, payload.sessionId);\n    }\n\n    // Update heartbeat, revive ended sessions, or auto-register orphans (#1870)\n    const session = ensureSession(payload.sessionId);\n    logger.debug(`[IngestRoutes] Metrics ingested from ${session?.displayName ?? payload.sessionId}`);\n    res.status(200).json({ accepted: true });\n  });\n\n  /**\n   * POST /api/ingest/session — Session lifecycle events.\n   */\n  router.post('/api/ingest/session', (req: Request, res: Response) => {\n    const payload = req.body as SessionEventPayload;\n    if (!payload?.sessionId || !payload.event) {\n      const received = payload ? Object.keys(payload) : [];\n      logger.warn('[IngestRoutes] Invalid session event payload', { received });\n      res.status(400).json({ error: 'Invalid payload', required: ['sessionId', 'event'], received });\n      return;\n    }\n    payload.sessionId = normalizeInput(payload.sessionId);\n\n    const now = new Date().toISOString();\n\n    switch (payload.event) {\n      case 'started': {\n        // Killed sessions stay dead; pending kills get finalized (#1870)\n        if (killedSessions.has(payload.sessionId)) break;\n        if (pendingKills.has(payload.sessionId)) { finalizePendingKill(payload.sessionId, payload.pid); break; }\n\n        const displayName = namePool.assign(payload.sessionId);\n        const color = namePool.getColor(payload.sessionId) ?? '#3b82f6';\n        const isAuthenticated = Boolean((res as any).locals?.tokenEntry);\n        sessions.set(payload.sessionId, {\n          sessionId: payload.sessionId, displayName, color,\n          pid: payload.pid, startedAt: payload.startedAt || now, lastHeartbeat: now,\n          status: 'active', isLeader: false, authenticated: isAuthenticated, kind: 'mcp',\n        });\n        logger.info('[IngestRoutes] Session registered', {\n          displayName, sessionId: payload.sessionId, pid: payload.pid, color,\n          activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length,\n        });\n        broadcasts.sessionBroadcast?.(sessions.get(payload.sessionId)!);\n        break;\n      }\n      case 'stopped': {\n        const existing = sessions.get(payload.sessionId);\n        if (existing) {\n          existing.status = 'ended';\n          existing.lastHeartbeat = now;\n          namePool.release(payload.sessionId);\n          logger.info('[IngestRoutes] Session stopped', {\n            displayName: existing.displayName, sessionId: payload.sessionId, pid: existing.pid,\n            activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n          });\n          broadcasts.sessionBroadcast?.(existing);\n        }\n        break;\n      }\n      case 'heartbeat': {\n        // Auto-register or update — heartbeat includes PID for recovery (#1870)\n        ensureSession(payload.sessionId, payload.pid);\n        break;\n      }\n    }\n\n    res.status(200).json({ ok: true });\n  });\n\n  /**\n   * GET /api/sessions — List all tracked sessions.\n   */\n  router.get('/api/sessions', async (_req: Request, res: Response) => {\n    // Server-side active filter — the frontend also filters, but ended sessions\n    // should never leave the API to prevent stale UI (#1870).\n    const localSessions = Array.from(sessions.values()).filter(s => s.status === 'active');\n    const currentPort = env.DOLLHOUSE_WEB_CONSOLE_PORT ?? 41715;\n\n    // Federate with the legacy port (3939) to show all sessions on the\n    // machine, including unauthenticated ones from pre-auth installs.\n    // Server-to-server avoids CORS restrictions (#1805).\n    if (currentPort !== 3939) {\n      try {\n        const controller = new AbortController();\n        const timeout = setTimeout(() => controller.abort(), LEGACY_FETCH_TIMEOUT_MS);\n        const legacyRes = await fetch('http://127.0.0.1:3939/api/sessions', {\n          signal: controller.signal,\n        });\n        clearTimeout(timeout);\n        if (legacyRes.ok) {\n          const legacyData = await legacyRes.json() as { sessions: SessionInfo[] };\n          const localIds = new Set(localSessions.map(s => s.sessionId));\n          for (const ls of (legacyData.sessions || [])) {\n            if (!localIds.has(ls.sessionId) && ls.status === 'active') {\n              localSessions.push({\n                ...ls,\n                authenticated: false,\n                kind: ls.kind || 'mcp',\n              });\n            }\n          }\n        }\n      } catch {\n        // Legacy instance not running or unreachable — that's fine\n      }\n    }\n\n    res.json({ sessions: localSessions });\n  });\n\n  /**\n   * POST /api/sessions/:sessionId/kill — Terminate a session's server process.\n   */\n  router.post('/api/sessions/:sessionId/kill', async (req: Request, res: Response) => {\n    const sessionId = req.params['sessionId'] as string;\n    const session = sessions.get(sessionId);\n\n    if (!session) {\n      // Session not in local Map — try proxying kill to legacy port (#1870)\n      const currentPort = env.DOLLHOUSE_WEB_CONSOLE_PORT ?? 41715;\n      if (currentPort !== 3939) {\n        try {\n          const controller = new AbortController();\n          const timeout = setTimeout(() => controller.abort(), LEGACY_FETCH_TIMEOUT_MS);\n          const proxyRes = await fetch(`http://127.0.0.1:3939/api/sessions/${encodeURIComponent(sessionId)}/kill`, {\n            method: 'POST',\n            signal: controller.signal,\n          });\n          clearTimeout(timeout);\n          if (proxyRes.ok) {\n            const data = await proxyRes.json();\n            res.json(data);\n            return;\n          }\n        } catch {\n          // Legacy instance not running — fall through to 404\n        }\n      }\n      logger.warn('[IngestRoutes] Kill requested for unknown session', { sessionId });\n      res.status(404).json({ error: 'Session not found', sessionId });\n      return;\n    }\n\n    if (!session.pid) {\n      // Auto-registered orphan with unknown PID — queue for deferred kill (#1870).\n      // The next heartbeat (every ~10s) carries the PID. ensureSession() will\n      // SIGTERM the process as soon as the PID arrives. Session is gone for good.\n      session.status = 'ended';\n      namePool.release(sessionId);\n      pendingKills.add(sessionId);\n      logger.info('[IngestRoutes] Queued deferred kill — waiting for PID via heartbeat', {\n        displayName: session.displayName, sessionId,\n      });\n      res.json({ ok: true, dismissed: session.displayName, reason: 'pending-kill' });\n      return;\n    }\n\n    // SIGTERM the process. Even if it fails (ESRCH = already dead, EPERM = not ours),\n    // mark the session as permanently killed so it never reappears (#1870).\n    let killed = false;\n    try {\n      process.kill(session.pid, 'SIGTERM');\n      killed = true;\n    } catch (err) {\n      const code = (err as NodeJS.ErrnoException).code;\n      if (code === 'ESRCH') {\n        killed = true; // process already dead — treat as successful kill\n      } else {\n        logger.error('[IngestRoutes] Failed to kill session', {\n          displayName: session.displayName, sessionId, pid: session.pid, error: (err as Error).message,\n        });\n        res.status(500).json({ error: 'Failed to kill session', sessionId, displayName: session.displayName, pid: session.pid, detail: (err as Error).message });\n        return;\n      }\n    }\n    session.status = 'ended';\n    namePool.release(sessionId);\n    killedSessions.add(sessionId);\n    logger.info('[IngestRoutes] Session killed', {\n      displayName: session.displayName, sessionId, pid: session.pid,\n      activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n    });\n    res.json({ ok: true, killed: session.displayName, pid: session.pid });\n  });\n\n  /** Mark stale active sessions as ended. */\n  function reapStaleSessions(now: number): void {\n    for (const [id, session] of sessions) {\n      if (session.status !== 'active') continue;\n      if (session.isLeader || session.kind === 'console') continue;\n      const age = now - new Date(session.lastHeartbeat).getTime();\n      if (age <= SESSION_STALE_MS) continue;\n      session.status = 'ended';\n      namePool.release(id);\n      logger.info('[IngestRoutes] Reaped stale session', {\n        displayName: session.displayName, sessionId: id, pid: session.pid,\n        lastHeartbeatAgo: `${Math.round(age / 1000)}s`,\n        activeSessions: Array.from(sessions.values()).filter(s => s.status === 'active').length - 1,\n      });\n      broadcasts.sessionBroadcast?.(session);\n    }\n  }\n\n  /** Delete ended sessions to bound memory (#1870). */\n  function purgeStaleEntries(now: number): void {\n    for (const [id, session] of sessions) {\n      if (session.status === 'ended' && now - new Date(session.lastHeartbeat).getTime() > ENDED_PURGE_MS) {\n        sessions.delete(id);\n      }\n    }\n  }\n\n  const reaperInterval = setInterval(() => {\n    const now = Date.now();\n    reapStaleSessions(now);\n    purgeStaleEntries(now);\n  }, REAPER_INTERVAL_MS);\n  reaperInterval.unref();\n\n  function getSessions(): SessionInfo[] {\n    return Array.from(sessions.values()).filter(s => s.status === 'active');\n  }\n\n  function registerLeaderSession(sessionId: string, pid: number): void {\n    const displayName = namePool.assign(sessionId, true);\n    const color = namePool.getColor(sessionId) ?? '#3b82f6';\n    sessions.set(sessionId, {\n      sessionId,\n      displayName,\n      color,\n      pid,\n      startedAt: new Date().toISOString(),\n      lastHeartbeat: new Date().toISOString(),\n      status: 'active',\n      isLeader: true,\n      authenticated: true,\n      kind: 'mcp',\n    });\n    logger.info('[IngestRoutes] Leader session registered', { displayName, sessionId, pid, color });\n  }\n\n  /**\n   * Register the web console itself as a session (#1805). Ensures the\n   * session indicator always shows at least one entry — the console the\n   * user is currently looking at.\n   */\n  function registerConsoleSession(): void {\n    const consoleId = `console-${process.pid}`;\n    if (sessions.has(consoleId)) return;\n    const displayName = 'Web Console';\n    sessions.set(consoleId, {\n      sessionId: consoleId,\n      displayName,\n      color: '#6366f1', // indigo — distinct from puppet greens/blues\n      pid: process.pid,\n      startedAt: new Date().toISOString(),\n      lastHeartbeat: new Date().toISOString(),\n      status: 'active',\n      isLeader: false,\n      authenticated: true,\n      kind: 'console',\n    });\n    logger.info('[IngestRoutes] Console session registered', { sessionId: consoleId, pid: process.pid });\n  }\n\n  return { router, getSessions, registerLeaderSession, registerConsoleSession };\n}\n"]}
|