@dollhousemcp/mcp-server 2.0.4 → 2.0.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (41) hide show
  1. package/CHANGELOG.md +8 -0
  2. package/dist/generated/version.d.ts +2 -2
  3. package/dist/generated/version.js +3 -3
  4. package/dist/handlers/mcp-aql/IntrospectionResolver.d.ts +2 -0
  5. package/dist/handlers/mcp-aql/IntrospectionResolver.d.ts.map +1 -1
  6. package/dist/handlers/mcp-aql/IntrospectionResolver.js +4 -1
  7. package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
  8. package/dist/handlers/mcp-aql/MCPAQLHandler.js +13 -7
  9. package/dist/handlers/mcp-aql/OperationRouter.d.ts +9 -0
  10. package/dist/handlers/mcp-aql/OperationRouter.d.ts.map +1 -1
  11. package/dist/handlers/mcp-aql/OperationRouter.js +53 -2
  12. package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
  13. package/dist/handlers/mcp-aql/OperationSchema.js +42 -3
  14. package/dist/index.js +34 -2
  15. package/dist/server/tools/MCPAQLTools.js +5 -1
  16. package/dist/utils/logger.d.ts +7 -0
  17. package/dist/utils/logger.d.ts.map +1 -1
  18. package/dist/utils/logger.js +13 -3
  19. package/dist/web/console/LeaderElection.d.ts +10 -0
  20. package/dist/web/console/LeaderElection.d.ts.map +1 -1
  21. package/dist/web/console/LeaderElection.js +44 -1
  22. package/dist/web/console/LeaderForwardingSink.d.ts +3 -1
  23. package/dist/web/console/LeaderForwardingSink.d.ts.map +1 -1
  24. package/dist/web/console/LeaderForwardingSink.js +24 -6
  25. package/dist/web/console/UnifiedConsole.d.ts.map +1 -1
  26. package/dist/web/console/UnifiedConsole.js +12 -3
  27. package/dist/web/public/app.js +24 -6
  28. package/dist/web/server.d.ts +3 -1
  29. package/dist/web/server.d.ts.map +1 -1
  30. package/dist/web/server.js +42 -17
  31. package/package.json +1 -1
  32. package/server.json +2 -2
  33. package/dist/constants/version.d.ts +0 -3
  34. package/dist/constants/version.d.ts.map +0 -1
  35. package/dist/constants/version.js +0 -4
  36. package/dist/logging/sinks/SSELogSink.d.ts +0 -35
  37. package/dist/logging/sinks/SSELogSink.d.ts.map +0 -1
  38. package/dist/logging/sinks/SSELogSink.js +0 -181
  39. package/dist/logging/viewer/viewerHtml.d.ts +0 -8
  40. package/dist/logging/viewer/viewerHtml.d.ts.map +0 -1
  41. package/dist/logging/viewer/viewerHtml.js +0 -204
@@ -13,6 +13,8 @@ import { ILogger, LogEntry } from '../types/ILogger.js';
13
13
  declare class MCPLogger implements ILogger {
14
14
  private logs;
15
15
  private isMCPConnected;
16
+ private minLevel;
17
+ private static readonly LEVEL_ORDER;
16
18
  private logListener?;
17
19
  addLogListener(fn: (entry: LogEntry) => void): () => void;
18
20
  private static readonly MAX_DEPTH;
@@ -25,6 +27,11 @@ declare class MCPLogger implements ILogger {
25
27
  * Call this after MCP connection is established to stop console output
26
28
  */
27
29
  setMCPConnected(): void;
30
+ /**
31
+ * Set minimum log level for console output.
32
+ * Entries below this level are still stored in memory but not printed.
33
+ */
34
+ setMinLevel(level: 'debug' | 'info' | 'warn' | 'error'): void;
28
35
  /**
29
36
  * Check if a field name contains sensitive patterns
30
37
  * Uses both exact matching and substring matching for better precision
@@ -1 +1 @@
1
- {"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../../src/utils/logger.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAC;AAGxD,cAAM,SAAU,YAAW,OAAO;IAChC,OAAO,CAAC,IAAI,CAAqC;IACjD,OAAO,CAAC,cAAc,CAAS;IAC/B,OAAO,CAAC,WAAW,CAAC,CAA4B;IAEhD,cAAc,CAAC,EAAE,EAAE,CAAC,KAAK,EAAE,QAAQ,KAAK,IAAI,GAAG,MAAM,IAAI;IAMzD,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAM;IAIvC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,oBAAoB,CAG1C;IAMF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,kBAAkB,CAIxC;IAGF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,iBAAiB,CAGvC;IAIF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,eAAe,CAGrC;IAMF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,0BAA0B,CAuB7C;IAEL;;OAEG;IACI,eAAe,IAAI,IAAI;IAI9B;;;;;OAKG;IACH,OAAO,CAAC,gBAAgB;IAkBxB;;;;;;;;OAQG;IACH,OAAO,CAAC,UAAU;IAgBlB;;;;;;OAMG;IACH,OAAO,CAAC,cAAc;IA6CtB;;;;;OAKG;IAEH,OAAO,CAAC,YAAY;IAWpB;;;;;OAKG;IAEH,OAAO,CAAC,eAAe;IAkCvB;;OAEG;IACH,OAAO,CAAC,GAAG;IA0CJ,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAIxC,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAIvC,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAIvC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAI/C;;OAEG;IACI,OAAO,CAAC,KAAK,SAAM,EAAE,KAAK,CAAC,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,QAAQ,EAAE;IAQlE;;OAEG;IACI,SAAS,IAAI,IAAI;CAGzB;AAGD,OAAO,EAAE,SAAS,EAAE,CAAC;AAGrB,eAAO,MAAM,MAAM,WAAkB,CAAC"}
1
+ {"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../../src/utils/logger.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAC;AAGxD,cAAM,SAAU,YAAW,OAAO;IAChC,OAAO,CAAC,IAAI,CAAqC;IACjD,OAAO,CAAC,cAAc,CAAS;IAC/B,OAAO,CAAC,QAAQ,CAAgD;IAChE,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,WAAW,CAA4C;IAC/E,OAAO,CAAC,WAAW,CAAC,CAA4B;IAEhD,cAAc,CAAC,EAAE,EAAE,CAAC,KAAK,EAAE,QAAQ,KAAK,IAAI,GAAG,MAAM,IAAI;IAMzD,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAM;IAIvC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,oBAAoB,CAG1C;IAMF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,kBAAkB,CAIxC;IAGF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,iBAAiB,CAGvC;IAIF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,eAAe,CAGrC;IAMF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,0BAA0B,CAuB7C;IAEL;;OAEG;IACI,eAAe,IAAI,IAAI;IAI9B;;;OAGG;IACI,WAAW,CAAC,KAAK,EAAE,OAAO,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,IAAI;IAIpE;;;;;OAKG;IACH,OAAO,CAAC,gBAAgB;IAkBxB;;;;;;;;OAQG;IACH,OAAO,CAAC,UAAU;IAgBlB;;;;;;OAMG;IACH,OAAO,CAAC,cAAc;IA6CtB;;;;;OAKG;IAEH,OAAO,CAAC,YAAY;IAWpB;;;;;OAKG;IAEH,OAAO,CAAC,eAAe;IAkCvB;;OAEG;IACH,OAAO,CAAC,GAAG;IA2CJ,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAIxC,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAIvC,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAIvC,KAAK,CAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,GAAG,GAAG,IAAI;IAI/C;;OAEG;IACI,OAAO,CAAC,KAAK,SAAM,EAAE,KAAK,CAAC,EAAE,QAAQ,CAAC,OAAO,CAAC,GAAG,QAAQ,EAAE;IAQlE;;OAEG;IACI,SAAS,IAAI,IAAI;CAGzB;AAGD,OAAO,EAAE,SAAS,EAAE,CAAC;AAGrB,eAAO,MAAM,MAAM,WAAkB,CAAC"}
@@ -13,6 +13,8 @@ import { EvictingQueue } from './EvictingQueue.js';
13
13
  class MCPLogger {
14
14
  logs = new EvictingQueue(1000);
15
15
  isMCPConnected = false;
16
+ minLevel = 'debug';
17
+ static LEVEL_ORDER = { debug: 0, info: 1, warn: 2, error: 3 };
16
18
  logListener;
17
19
  addLogListener(fn) {
18
20
  this.logListener = fn;
@@ -69,6 +71,13 @@ class MCPLogger {
69
71
  setMCPConnected() {
70
72
  this.isMCPConnected = true;
71
73
  }
74
+ /**
75
+ * Set minimum log level for console output.
76
+ * Entries below this level are still stored in memory but not printed.
77
+ */
78
+ setMinLevel(level) {
79
+ this.minLevel = level;
80
+ }
72
81
  /**
73
82
  * Check if a field name contains sensitive patterns
74
83
  * Uses both exact matching and substring matching for better precision
@@ -227,11 +236,12 @@ class MCPLogger {
227
236
  // Bounded FIFO eviction — EvictingQueue handles capacity
228
237
  this.logs.push(entry);
229
238
  this.logListener?.(entry);
230
- // Only write to console during initialization
239
+ // Only write to console during initialization, respecting minimum level
231
240
  if (!this.isMCPConnected) {
232
241
  // Check NODE_ENV inside the method to ensure it's evaluated at runtime
233
242
  const isTest = process.env.NODE_ENV === 'test';
234
- if (!isTest) {
243
+ const meetsLevel = MCPLogger.LEVEL_ORDER[level] >= MCPLogger.LEVEL_ORDER[this.minLevel];
244
+ if (!isTest && meetsLevel) {
235
245
  const prefix = `[${entry.timestamp.toISOString()}] [${level.toUpperCase()}]`;
236
246
  // Security fix: Use sanitized message to prevent sensitive information disclosure
237
247
  // Both message and data are sanitized before any output
@@ -286,4 +296,4 @@ class MCPLogger {
286
296
  export { MCPLogger };
287
297
  // Singleton instance for convenience
288
298
  export const logger = new MCPLogger();
289
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/utils/logger.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MCP-safe logger that avoids writing to stdout/stderr during protocol communication\n *\n * In MCP servers, stdout and stderr are reserved for JSON-RPC protocol messages.\n * Any non-protocol output will cause \"Unexpected token\" errors in the MCP client.\n *\n * This logger:\n * - Writes to stderr ONLY during server initialization (before MCP connection)\n * - Stores all logs in memory during runtime\n * - Provides methods to retrieve logs via MCP tools if needed\n */\n\nimport { ILogger, LogEntry } from '../types/ILogger.js';\nimport { EvictingQueue } from './EvictingQueue.js';\n\nclass MCPLogger implements ILogger {\n  private logs = new EvictingQueue<LogEntry>(1000);\n  private isMCPConnected = false;\n  private logListener?: (entry: LogEntry) => void;\n\n  addLogListener(fn: (entry: LogEntry) => void): () => void {\n    this.logListener = fn;\n    return () => { this.logListener = undefined; };\n  }\n  \n  // Performance: Maximum depth for object sanitization\n  private static readonly MAX_DEPTH = 10;\n  \n  // Sensitive field patterns with different matching strategies\n  // Exact match patterns - must match the entire field name\n  private static readonly EXACT_MATCH_PATTERNS = [\n    'password', 'token', 'secret', 'key', 'authorization',\n    'auth', 'credential', 'private', 'session', 'cookie'\n  ];\n  \n  // Substring match patterns - can appear anywhere in field name\n  // These are pattern names for detection, not actual sensitive values\n  // Building from character codes to avoid CodeQL false positives\n  // lgtm[js/clear-text-logging]\n  private static readonly SUBSTRING_PATTERNS = [\n    'api_key', 'apikey', 'access_token', 'refresh_token',\n    'client_secret', 'client_id', 'bearer',\n    String.fromCodePoint(111, 97, 117, 116, 104)  // 'oauth' - char codes prevent CodeQL false positive\n  ];\n  \n  // Performance optimization: Pre-compiled regex patterns\n  private static readonly EXACT_MATCH_REGEX = new RegExp(\n    `^(${MCPLogger.EXACT_MATCH_PATTERNS.join('|')})$`,\n    'i'\n  );\n  \n  // Use partial word boundaries - start boundary but allow suffixes\n  // This catches \"oauth_token\" and \"api_keys\" but not \"authentication\"\n  private static readonly SUBSTRING_REGEX = new RegExp(\n    `(^|[^a-zA-Z])(${MCPLogger.SUBSTRING_PATTERNS.join('|')})`,\n    'i'\n  );\n  \n  // Patterns for detecting sensitive data in log messages\n  // These are detection patterns used to IDENTIFY and REDACT sensitive data, not actual credentials\n  // Using indirect construction to avoid CodeQL false positive detection\n  // lgtm[js/clear-text-logging]\n  private static readonly MESSAGE_SENSITIVE_PATTERNS = (() => {\n    // Build patterns without literal sensitive strings\n    const patterns: RegExp[] = [];\n    \n    // Standard patterns\n    patterns.push(/\\b(token|password|secret|key|auth|bearer)\\s*[:=]\\s*[\\w\\-_\\.]+/gi);\n    patterns.push(/\\b(api[_-]?key)\\s*[:=]\\s*[\\w\\-_\\.]+/gi);\n    \n    // Patterns built indirectly to avoid detection\n    // lgtm[js/clear-text-logging]\n    patterns.push(new RegExp(`\\\\b(${['access', 'token'].join('[_-]?')})\\\\s*[:=]\\\\s*[\\\\w\\\\-_\\\\.]+`, 'gi'));\n    patterns.push(/\\b(refresh[_-]?token)\\s*[:=]\\s*[\\w\\-_\\.]+/gi);\n    \n    // lgtm[js/clear-text-logging]\n    patterns.push(new RegExp(`\\\\b(${['client', 'secret'].join('[_-]?')})\\\\s*[:=]\\\\s*[\\\\w\\\\-_\\\\.]+`, 'gi'));\n    patterns.push(new RegExp(`\\\\b(${['client', 'id'].join('[_-]?')})\\\\s*[:=]\\\\s*[\\\\w\\\\-_\\\\.]+`, 'gi'));\n    patterns.push(/Bearer\\s+[\\w\\-_\\.]+/gi);\n    \n    // lgtm[js/clear-text-logging]\n    const apiPattern = ['sk', 'pk', String.fromCodePoint(97, 112, 105)].join('|'); // 'api' - char codes prevent CodeQL false positive\n    patterns.push(new RegExp(`\\\\b(${apiPattern})[-_][\\\\w\\\\-]+`, 'gi'));\n    \n    return patterns;\n  })();\n  \n  /**\n   * Call this after MCP connection is established to stop console output\n   */\n  public setMCPConnected(): void {\n    this.isMCPConnected = true;\n  }\n\n  /**\n   * Check if a field name contains sensitive patterns\n   * Uses both exact matching and substring matching for better precision\n   * @param fieldName - The field name to check\n   * @returns true if the field name matches sensitive patterns\n   */\n  private isSensitiveField(fieldName: string): boolean {\n    // First check exact matches (e.g., \"password\" but not \"password_hint\")\n    if (MCPLogger.EXACT_MATCH_REGEX.test(fieldName)) {\n      return true;\n    }\n    \n    // Then check substring patterns (e.g., \"api_key\", \"access_token\", \"oauth_token\")\n    // Also check if the field name itself contains these patterns\n    const lowerFieldName = fieldName.toLowerCase();\n    for (const pattern of MCPLogger.SUBSTRING_PATTERNS) {\n      if (lowerFieldName.includes(pattern)) {\n        return true;\n      }\n    }\n    \n    return false;\n  }\n\n  /**\n   * Safely assign a value, ensuring sensitive data is never exposed\n   * This function makes it explicit to CodeQL that sensitive values are replaced\n   * @param key - The object key\n   * @param value - The value to potentially sanitize\n   * @param depth - Current recursion depth for performance protection\n   * @param seen - Set of seen objects to prevent circular references\n   * @returns Safe value that can be logged\n   */\n  private safeAssign(key: string, value: any, depth: number, seen: WeakSet<any>): any {\n    // Explicitly check if this is a sensitive field BEFORE any assignment\n    if (this.isSensitiveField(key)) {\n      // Return a constant redacted string - no sensitive data flows through\n      return '[REDACTED]';\n    }\n    \n    // For non-sensitive fields, recursively sanitize if needed\n    if (typeof value === 'object' && value !== null) {\n      return this.sanitizeObject(value, depth, seen);\n    }\n    \n    // Primitive non-sensitive values are safe to return\n    return value;\n  }\n\n  /**\n   * Sanitize an object or array recursively with performance optimizations\n   * @param obj - Object or array to sanitize\n   * @param depth - Current recursion depth (defaults to 0)\n   * @param seen - Set of seen objects to detect circular references\n   * @returns Sanitized copy with sensitive fields redacted\n   */\n  private sanitizeObject(obj: any, depth: number = 0, seen?: WeakSet<any>): any {\n    // Handle null/undefined\n    if (obj == null) return obj;\n    \n    // Handle non-objects (primitives)\n    if (typeof obj !== 'object') return obj;\n    \n    // Performance: Depth limiting to prevent stack overflow\n    if (depth >= MCPLogger.MAX_DEPTH) {\n      return '[DEEP_OBJECT_TRUNCATED]';\n    }\n    \n    // Performance: Circular reference detection\n    if (!seen) {\n      seen = new WeakSet();\n    }\n    \n    // Check for circular references\n    if (seen.has(obj)) {\n      return '[CIRCULAR_REFERENCE]';\n    }\n    \n    // Mark this object as seen\n    seen.add(obj);\n    \n    // Handle arrays\n    if (Array.isArray(obj)) {\n      return obj.map(item => {\n        if (typeof item === 'object' && item !== null) {\n          return this.sanitizeObject(item, depth + 1, seen);\n        }\n        return item;\n      });\n    }\n    \n    // Handle objects - use safe assignment for each field\n    const sanitized: any = {};\n    for (const [key, value] of Object.entries(obj)) {\n      // Use safe assignment which checks sensitivity and returns safe values\n      sanitized[key] = this.safeAssign(key, value, depth + 1, seen);\n    }\n    \n    return sanitized;\n  }\n\n  /**\n   * Sanitize sensitive data before logging\n   * Security fix: Prevents exposure of OAuth tokens, API keys, passwords, etc.\n   * @param data - Data to sanitize (can be any type)\n   * @returns Sanitized copy with sensitive fields replaced with '[REDACTED]'\n   */\n  // lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it\n  private sanitizeData(data: any): any {\n    // Fast path for null/undefined\n    if (data == null) return data;\n    \n    // Fast path for primitives\n    if (typeof data !== 'object') return data;\n    \n    // Sanitize objects and arrays\n    return this.sanitizeObject(data);\n  }\n  \n  /**\n   * Sanitize sensitive information from log messages\n   * Security fix: Prevents exposure of credentials that may be embedded in message strings\n   * @param message - The log message to sanitize\n   * @returns Sanitized message with sensitive data replaced with '[REDACTED]'\n   */\n  // lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it\n  private sanitizeMessage(message: string): string {\n    if (!message || typeof message !== 'string') {\n      return message;\n    }\n    \n    let sanitized = message;\n    \n    // Apply each sensitive pattern to detect and redact sensitive data\n    MCPLogger.MESSAGE_SENSITIVE_PATTERNS.forEach(pattern => {\n      sanitized = sanitized.replace(pattern, (match) => {\n        // For key=value patterns, preserve the key but redact the value\n        if (match.includes('=') || match.includes(':')) {\n          const separator = match.includes('=') ? '=' : ':';\n          const parts = match.split(separator);\n          if (parts.length >= 2) {\n            return `${parts[0]}${separator}[REDACTED]`;\n          }\n        }\n        // For Bearer tokens or standalone sensitive values\n        if (match.toLowerCase().startsWith('bearer')) {\n          return 'Bearer [REDACTED]';\n        }\n        // For API keys like sk-xxxxx\n        if (/^(sk|pk|api)[-_]/i.test(match)) {\n          return match.substring(0, 3) + '[REDACTED]';\n        }\n        // Default: redact the entire match\n        return '[REDACTED]';\n      });\n    });\n    \n    return sanitized;\n  }\n  \n  /**\n   * Internal logging method\n   */\n  private log(level: LogEntry['level'], message: string, data?: any): void {\n    // Sanitize both message and data to prevent sensitive info exposure\n    const sanitizedMessage = this.sanitizeMessage(message);\n    const sanitizedData = this.sanitizeData(data);\n    \n    const entry: LogEntry = {\n      timestamp: new Date(),\n      level,\n      message: sanitizedMessage,  // Store sanitized message\n      data: sanitizedData\n    };\n    \n    // Bounded FIFO eviction — EvictingQueue handles capacity\n    this.logs.push(entry);\n    this.logListener?.(entry);\n\n    // Only write to console during initialization\n    if (!this.isMCPConnected) {\n      // Check NODE_ENV inside the method to ensure it's evaluated at runtime\n      const isTest = process.env.NODE_ENV === 'test';\n      if (!isTest) {\n        const prefix = `[${entry.timestamp.toISOString()}] [${level.toUpperCase()}]`;\n        // Security fix: Use sanitized message to prevent sensitive information disclosure\n        // Both message and data are sanitized before any output\n        const safeMessage = `${prefix} ${sanitizedMessage}`;\n        \n        // During initialization, we can use console\n        if (level === 'error') {\n          // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()\n          console.error(safeMessage);\n        } else if (level === 'warn') {\n          // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()\n          console.warn(safeMessage);\n        } else {\n          // For MCP, even during init, avoid stdout for info/debug\n          // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()\n          console.error(safeMessage);\n        }\n      }\n    }\n  }\n  \n  public debug(message: string, data?: any): void {\n    this.log('debug', message, data);\n  }\n  \n  public info(message: string, data?: any): void {\n    this.log('info', message, data);\n  }\n  \n  public warn(message: string, data?: any): void {\n    this.log('warn', message, data);\n  }\n  \n  public error(message: string, data?: any): void {\n    this.log('error', message, data);\n  }\n  \n  /**\n   * Get recent logs (for MCP tools to retrieve)\n   */\n  public getLogs(count = 100, level?: LogEntry['level']): LogEntry[] {\n    let filtered: readonly LogEntry[] = this.logs.toArray();\n    if (level) {\n      filtered = filtered.filter(log => log.level === level);\n    }\n    return filtered.slice(-count);\n  }\n  \n  /**\n   * Clear logs\n   */\n  public clearLogs(): void {\n    this.logs.clear();\n  }\n}\n\n// Export class for testing/extension\nexport { MCPLogger };\n\n// Singleton instance for convenience\nexport const logger = new MCPLogger();"]}
299
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../../src/utils/logger.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MCP-safe logger that avoids writing to stdout/stderr during protocol communication\n *\n * In MCP servers, stdout and stderr are reserved for JSON-RPC protocol messages.\n * Any non-protocol output will cause \"Unexpected token\" errors in the MCP client.\n *\n * This logger:\n * - Writes to stderr ONLY during server initialization (before MCP connection)\n * - Stores all logs in memory during runtime\n * - Provides methods to retrieve logs via MCP tools if needed\n */\n\nimport { ILogger, LogEntry } from '../types/ILogger.js';\nimport { EvictingQueue } from './EvictingQueue.js';\n\nclass MCPLogger implements ILogger {\n  private logs = new EvictingQueue<LogEntry>(1000);\n  private isMCPConnected = false;\n  private minLevel: 'debug' | 'info' | 'warn' | 'error' = 'debug';\n  private static readonly LEVEL_ORDER = { debug: 0, info: 1, warn: 2, error: 3 };\n  private logListener?: (entry: LogEntry) => void;\n\n  addLogListener(fn: (entry: LogEntry) => void): () => void {\n    this.logListener = fn;\n    return () => { this.logListener = undefined; };\n  }\n  \n  // Performance: Maximum depth for object sanitization\n  private static readonly MAX_DEPTH = 10;\n  \n  // Sensitive field patterns with different matching strategies\n  // Exact match patterns - must match the entire field name\n  private static readonly EXACT_MATCH_PATTERNS = [\n    'password', 'token', 'secret', 'key', 'authorization',\n    'auth', 'credential', 'private', 'session', 'cookie'\n  ];\n  \n  // Substring match patterns - can appear anywhere in field name\n  // These are pattern names for detection, not actual sensitive values\n  // Building from character codes to avoid CodeQL false positives\n  // lgtm[js/clear-text-logging]\n  private static readonly SUBSTRING_PATTERNS = [\n    'api_key', 'apikey', 'access_token', 'refresh_token',\n    'client_secret', 'client_id', 'bearer',\n    String.fromCodePoint(111, 97, 117, 116, 104)  // 'oauth' - char codes prevent CodeQL false positive\n  ];\n  \n  // Performance optimization: Pre-compiled regex patterns\n  private static readonly EXACT_MATCH_REGEX = new RegExp(\n    `^(${MCPLogger.EXACT_MATCH_PATTERNS.join('|')})$`,\n    'i'\n  );\n  \n  // Use partial word boundaries - start boundary but allow suffixes\n  // This catches \"oauth_token\" and \"api_keys\" but not \"authentication\"\n  private static readonly SUBSTRING_REGEX = new RegExp(\n    `(^|[^a-zA-Z])(${MCPLogger.SUBSTRING_PATTERNS.join('|')})`,\n    'i'\n  );\n  \n  // Patterns for detecting sensitive data in log messages\n  // These are detection patterns used to IDENTIFY and REDACT sensitive data, not actual credentials\n  // Using indirect construction to avoid CodeQL false positive detection\n  // lgtm[js/clear-text-logging]\n  private static readonly MESSAGE_SENSITIVE_PATTERNS = (() => {\n    // Build patterns without literal sensitive strings\n    const patterns: RegExp[] = [];\n    \n    // Standard patterns\n    patterns.push(/\\b(token|password|secret|key|auth|bearer)\\s*[:=]\\s*[\\w\\-_\\.]+/gi);\n    patterns.push(/\\b(api[_-]?key)\\s*[:=]\\s*[\\w\\-_\\.]+/gi);\n    \n    // Patterns built indirectly to avoid detection\n    // lgtm[js/clear-text-logging]\n    patterns.push(new RegExp(`\\\\b(${['access', 'token'].join('[_-]?')})\\\\s*[:=]\\\\s*[\\\\w\\\\-_\\\\.]+`, 'gi'));\n    patterns.push(/\\b(refresh[_-]?token)\\s*[:=]\\s*[\\w\\-_\\.]+/gi);\n    \n    // lgtm[js/clear-text-logging]\n    patterns.push(new RegExp(`\\\\b(${['client', 'secret'].join('[_-]?')})\\\\s*[:=]\\\\s*[\\\\w\\\\-_\\\\.]+`, 'gi'));\n    patterns.push(new RegExp(`\\\\b(${['client', 'id'].join('[_-]?')})\\\\s*[:=]\\\\s*[\\\\w\\\\-_\\\\.]+`, 'gi'));\n    patterns.push(/Bearer\\s+[\\w\\-_\\.]+/gi);\n    \n    // lgtm[js/clear-text-logging]\n    const apiPattern = ['sk', 'pk', String.fromCodePoint(97, 112, 105)].join('|'); // 'api' - char codes prevent CodeQL false positive\n    patterns.push(new RegExp(`\\\\b(${apiPattern})[-_][\\\\w\\\\-]+`, 'gi'));\n    \n    return patterns;\n  })();\n  \n  /**\n   * Call this after MCP connection is established to stop console output\n   */\n  public setMCPConnected(): void {\n    this.isMCPConnected = true;\n  }\n\n  /**\n   * Set minimum log level for console output.\n   * Entries below this level are still stored in memory but not printed.\n   */\n  public setMinLevel(level: 'debug' | 'info' | 'warn' | 'error'): void {\n    this.minLevel = level;\n  }\n\n  /**\n   * Check if a field name contains sensitive patterns\n   * Uses both exact matching and substring matching for better precision\n   * @param fieldName - The field name to check\n   * @returns true if the field name matches sensitive patterns\n   */\n  private isSensitiveField(fieldName: string): boolean {\n    // First check exact matches (e.g., \"password\" but not \"password_hint\")\n    if (MCPLogger.EXACT_MATCH_REGEX.test(fieldName)) {\n      return true;\n    }\n    \n    // Then check substring patterns (e.g., \"api_key\", \"access_token\", \"oauth_token\")\n    // Also check if the field name itself contains these patterns\n    const lowerFieldName = fieldName.toLowerCase();\n    for (const pattern of MCPLogger.SUBSTRING_PATTERNS) {\n      if (lowerFieldName.includes(pattern)) {\n        return true;\n      }\n    }\n    \n    return false;\n  }\n\n  /**\n   * Safely assign a value, ensuring sensitive data is never exposed\n   * This function makes it explicit to CodeQL that sensitive values are replaced\n   * @param key - The object key\n   * @param value - The value to potentially sanitize\n   * @param depth - Current recursion depth for performance protection\n   * @param seen - Set of seen objects to prevent circular references\n   * @returns Safe value that can be logged\n   */\n  private safeAssign(key: string, value: any, depth: number, seen: WeakSet<any>): any {\n    // Explicitly check if this is a sensitive field BEFORE any assignment\n    if (this.isSensitiveField(key)) {\n      // Return a constant redacted string - no sensitive data flows through\n      return '[REDACTED]';\n    }\n    \n    // For non-sensitive fields, recursively sanitize if needed\n    if (typeof value === 'object' && value !== null) {\n      return this.sanitizeObject(value, depth, seen);\n    }\n    \n    // Primitive non-sensitive values are safe to return\n    return value;\n  }\n\n  /**\n   * Sanitize an object or array recursively with performance optimizations\n   * @param obj - Object or array to sanitize\n   * @param depth - Current recursion depth (defaults to 0)\n   * @param seen - Set of seen objects to detect circular references\n   * @returns Sanitized copy with sensitive fields redacted\n   */\n  private sanitizeObject(obj: any, depth: number = 0, seen?: WeakSet<any>): any {\n    // Handle null/undefined\n    if (obj == null) return obj;\n    \n    // Handle non-objects (primitives)\n    if (typeof obj !== 'object') return obj;\n    \n    // Performance: Depth limiting to prevent stack overflow\n    if (depth >= MCPLogger.MAX_DEPTH) {\n      return '[DEEP_OBJECT_TRUNCATED]';\n    }\n    \n    // Performance: Circular reference detection\n    if (!seen) {\n      seen = new WeakSet();\n    }\n    \n    // Check for circular references\n    if (seen.has(obj)) {\n      return '[CIRCULAR_REFERENCE]';\n    }\n    \n    // Mark this object as seen\n    seen.add(obj);\n    \n    // Handle arrays\n    if (Array.isArray(obj)) {\n      return obj.map(item => {\n        if (typeof item === 'object' && item !== null) {\n          return this.sanitizeObject(item, depth + 1, seen);\n        }\n        return item;\n      });\n    }\n    \n    // Handle objects - use safe assignment for each field\n    const sanitized: any = {};\n    for (const [key, value] of Object.entries(obj)) {\n      // Use safe assignment which checks sensitivity and returns safe values\n      sanitized[key] = this.safeAssign(key, value, depth + 1, seen);\n    }\n    \n    return sanitized;\n  }\n\n  /**\n   * Sanitize sensitive data before logging\n   * Security fix: Prevents exposure of OAuth tokens, API keys, passwords, etc.\n   * @param data - Data to sanitize (can be any type)\n   * @returns Sanitized copy with sensitive fields replaced with '[REDACTED]'\n   */\n  // lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it\n  private sanitizeData(data: any): any {\n    // Fast path for null/undefined\n    if (data == null) return data;\n    \n    // Fast path for primitives\n    if (typeof data !== 'object') return data;\n    \n    // Sanitize objects and arrays\n    return this.sanitizeObject(data);\n  }\n  \n  /**\n   * Sanitize sensitive information from log messages\n   * Security fix: Prevents exposure of credentials that may be embedded in message strings\n   * @param message - The log message to sanitize\n   * @returns Sanitized message with sensitive data replaced with '[REDACTED]'\n   */\n  // lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it\n  private sanitizeMessage(message: string): string {\n    if (!message || typeof message !== 'string') {\n      return message;\n    }\n    \n    let sanitized = message;\n    \n    // Apply each sensitive pattern to detect and redact sensitive data\n    MCPLogger.MESSAGE_SENSITIVE_PATTERNS.forEach(pattern => {\n      sanitized = sanitized.replace(pattern, (match) => {\n        // For key=value patterns, preserve the key but redact the value\n        if (match.includes('=') || match.includes(':')) {\n          const separator = match.includes('=') ? '=' : ':';\n          const parts = match.split(separator);\n          if (parts.length >= 2) {\n            return `${parts[0]}${separator}[REDACTED]`;\n          }\n        }\n        // For Bearer tokens or standalone sensitive values\n        if (match.toLowerCase().startsWith('bearer')) {\n          return 'Bearer [REDACTED]';\n        }\n        // For API keys like sk-xxxxx\n        if (/^(sk|pk|api)[-_]/i.test(match)) {\n          return match.substring(0, 3) + '[REDACTED]';\n        }\n        // Default: redact the entire match\n        return '[REDACTED]';\n      });\n    });\n    \n    return sanitized;\n  }\n  \n  /**\n   * Internal logging method\n   */\n  private log(level: LogEntry['level'], message: string, data?: any): void {\n    // Sanitize both message and data to prevent sensitive info exposure\n    const sanitizedMessage = this.sanitizeMessage(message);\n    const sanitizedData = this.sanitizeData(data);\n    \n    const entry: LogEntry = {\n      timestamp: new Date(),\n      level,\n      message: sanitizedMessage,  // Store sanitized message\n      data: sanitizedData\n    };\n    \n    // Bounded FIFO eviction — EvictingQueue handles capacity\n    this.logs.push(entry);\n    this.logListener?.(entry);\n\n    // Only write to console during initialization, respecting minimum level\n    if (!this.isMCPConnected) {\n      // Check NODE_ENV inside the method to ensure it's evaluated at runtime\n      const isTest = process.env.NODE_ENV === 'test';\n      const meetsLevel = MCPLogger.LEVEL_ORDER[level] >= MCPLogger.LEVEL_ORDER[this.minLevel];\n      if (!isTest && meetsLevel) {\n        const prefix = `[${entry.timestamp.toISOString()}] [${level.toUpperCase()}]`;\n        // Security fix: Use sanitized message to prevent sensitive information disclosure\n        // Both message and data are sanitized before any output\n        const safeMessage = `${prefix} ${sanitizedMessage}`;\n        \n        // During initialization, we can use console\n        if (level === 'error') {\n          // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()\n          console.error(safeMessage);\n        } else if (level === 'warn') {\n          // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()\n          console.warn(safeMessage);\n        } else {\n          // For MCP, even during init, avoid stdout for info/debug\n          // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage()\n          console.error(safeMessage);\n        }\n      }\n    }\n  }\n  \n  public debug(message: string, data?: any): void {\n    this.log('debug', message, data);\n  }\n  \n  public info(message: string, data?: any): void {\n    this.log('info', message, data);\n  }\n  \n  public warn(message: string, data?: any): void {\n    this.log('warn', message, data);\n  }\n  \n  public error(message: string, data?: any): void {\n    this.log('error', message, data);\n  }\n  \n  /**\n   * Get recent logs (for MCP tools to retrieve)\n   */\n  public getLogs(count = 100, level?: LogEntry['level']): LogEntry[] {\n    let filtered: readonly LogEntry[] = this.logs.toArray();\n    if (level) {\n      filtered = filtered.filter(log => log.level === level);\n    }\n    return filtered.slice(-count);\n  }\n  \n  /**\n   * Clear logs\n   */\n  public clearLogs(): void {\n    this.logs.clear();\n  }\n}\n\n// Export class for testing/extension\nexport { MCPLogger };\n\n// Singleton instance for convenience\nexport const logger = new MCPLogger();"]}
@@ -74,6 +74,16 @@ export declare function deleteLeaderLock(): Promise<void>;
74
74
  * @returns Election result with role and leader info
75
75
  */
76
76
  export declare function electLeader(sessionId: string, port: number): Promise<ElectionResult>;
77
+ /**
78
+ * Probe whether the leader's web console is reachable.
79
+ * Returns true if the leader's ingest endpoint responds, false otherwise.
80
+ */
81
+ export declare function isLeaderWebConsoleReachable(leaderInfo: ConsoleLeaderInfo): Promise<boolean>;
82
+ /**
83
+ * Force claim leadership by deleting the existing lock and claiming.
84
+ * Used when the existing leader is alive but not running a web console.
85
+ */
86
+ export declare function forceClaimLeadership(sessionId: string, port: number): Promise<ElectionResult>;
77
87
  /**
78
88
  * Start the leader heartbeat loop.
79
89
  * Updates the lock file every HEARTBEAT_INTERVAL_MS so followers know the leader is alive.
@@ -1 +1 @@
1
- {"version":3,"file":"LeaderElection.d.ts","sourceRoot":"","sources":["../../../src/web/console/LeaderElection.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;GAiBG;AAuBH;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,OAAO,EAAE,MAAM,CAAC;IAChB,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,QAAQ,GAAG,UAAU,CAAC;IAC5B,sEAAsE;IACtE,UAAU,EAAE,iBAAiB,CAAC;CAC/B;AAED;;;GAGG;AACH,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAOnD;AAED;;;GAGG;AACH,wBAAsB,cAAc,IAAI,OAAO,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAWxE;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,iBAAiB,GAAG,OAAO,CAM5D;AAED;;;;;;;;GAQG;AACH,wBAAsB,eAAe,CAAC,IAAI,EAAE,iBAAiB,GAAG,OAAO,CAAC,OAAO,CAAC,CAe/E;AAED;;GAEG;AACH,wBAAsB,gBAAgB,IAAI,OAAO,CAAC,IAAI,CAAC,CAEtD;AAED;;;;;;;;;GASG;AACH,wBAAsB,WAAW,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,CAyD1F;AAED;;;;;GAKG;AACH,wBAAgB,cAAc,CAAC,IAAI,EAAE,iBAAiB,GAAG,MAAM,IAAI,CAgBlE;AAED;;;GAGG;AACH,wBAAgB,qBAAqB,IAAI,IAAI,CAK5C"}
1
+ {"version":3,"file":"LeaderElection.d.ts","sourceRoot":"","sources":["../../../src/web/console/LeaderElection.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;GAiBG;AAuBH;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,OAAO,EAAE,MAAM,CAAC;IAChB,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,QAAQ,GAAG,UAAU,CAAC;IAC5B,sEAAsE;IACtE,UAAU,EAAE,iBAAiB,CAAC;CAC/B;AAED;;;GAGG;AACH,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAOnD;AAED;;;GAGG;AACH,wBAAsB,cAAc,IAAI,OAAO,CAAC,iBAAiB,GAAG,IAAI,CAAC,CAWxE;AAED;;GAEG;AACH,wBAAgB,WAAW,CAAC,IAAI,EAAE,iBAAiB,GAAG,OAAO,CAM5D;AAED;;;;;;;;GAQG;AACH,wBAAsB,eAAe,CAAC,IAAI,EAAE,iBAAiB,GAAG,OAAO,CAAC,OAAO,CAAC,CAe/E;AAED;;GAEG;AACH,wBAAsB,gBAAgB,IAAI,OAAO,CAAC,IAAI,CAAC,CAEtD;AAED;;;;;;;;;GASG;AACH,wBAAsB,WAAW,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,CAyD1F;AAED;;;GAGG;AACH,wBAAsB,2BAA2B,CAAC,UAAU,EAAE,iBAAiB,GAAG,OAAO,CAAC,OAAO,CAAC,CAYjG;AAED;;;GAGG;AACH,wBAAsB,oBAAoB,CAAC,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,CAAC,CAuBnG;AAED;;;;;GAKG;AACH,wBAAgB,cAAc,CAAC,IAAI,EAAE,iBAAiB,GAAG,MAAM,IAAI,CAgBlE;AAED;;;GAGG;AACH,wBAAgB,qBAAqB,IAAI,IAAI,CAK5C"}
@@ -170,6 +170,49 @@ export async function electLeader(sessionId, port) {
170
170
  const actualLeader = await readLeaderLock();
171
171
  return { role: 'follower', leaderInfo: actualLeader ?? retryInfo };
172
172
  }
173
+ /**
174
+ * Probe whether the leader's web console is reachable.
175
+ * Returns true if the leader's ingest endpoint responds, false otherwise.
176
+ */
177
+ export async function isLeaderWebConsoleReachable(leaderInfo) {
178
+ try {
179
+ const controller = new AbortController();
180
+ const timeout = setTimeout(() => controller.abort(), 2_000);
181
+ const res = await fetch(`http://127.0.0.1:${leaderInfo.port}/api/logs/stats`, {
182
+ signal: controller.signal,
183
+ });
184
+ clearTimeout(timeout);
185
+ return res.ok;
186
+ }
187
+ catch {
188
+ return false;
189
+ }
190
+ }
191
+ /**
192
+ * Force claim leadership by deleting the existing lock and claiming.
193
+ * Used when the existing leader is alive but not running a web console.
194
+ */
195
+ export async function forceClaimLeadership(sessionId, port) {
196
+ logger.info('[LeaderElection] Forcing leadership takeover — existing leader has no web console');
197
+ await deleteLeaderLock();
198
+ const now = new Date().toISOString();
199
+ const myInfo = {
200
+ version: LOCK_VERSION,
201
+ pid: process.pid,
202
+ port,
203
+ sessionId: UnicodeValidator.normalize(sessionId).normalizedContent,
204
+ startedAt: now,
205
+ heartbeat: now,
206
+ };
207
+ const claimed = await claimLeadership(myInfo);
208
+ if (claimed) {
209
+ logger.info('[LeaderElection] Forced leadership claimed', { sessionId, port, pid: process.pid });
210
+ return { role: 'leader', leaderInfo: myInfo };
211
+ }
212
+ // Failed — fall back to follower
213
+ const winner = await readLeaderLock();
214
+ return { role: 'follower', leaderInfo: winner ?? myInfo };
215
+ }
173
216
  /**
174
217
  * Start the leader heartbeat loop.
175
218
  * Updates the lock file every HEARTBEAT_INTERVAL_MS so followers know the leader is alive.
@@ -202,4 +245,4 @@ export function registerLeaderCleanup() {
202
245
  process.once('SIGTERM', cleanup);
203
246
  process.once('SIGINT', cleanup);
204
247
  }
205
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LeaderElection.js","sourceRoot":"","sources":["../../../src/web/console/LeaderElection.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Leader election for the unified web console.\n *\n * When multiple MCP server instances run concurrently, only one should host\n * the web console (the \"leader\"). Others become \"followers\" that forward\n * events to the leader. This module handles:\n *\n * 1. Reading/writing a leader lock file at ~/.dollhouse/run/console-leader.lock\n * 2. Atomic claim via temp+rename to prevent TOCTOU races\n * 3. PID-based stale detection (signal-0 liveness check)\n * 4. Heartbeat updates (10s interval) so followers can detect hung leaders\n * 5. Cleanup on process exit\n *\n * The port 3939 binding is the ultimate tiebreaker: even if two processes\n * both write the lock file, only one can bind the port.\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport { homedir } from 'node:os';\nimport { join } from 'node:path';\nimport { mkdir, readFile, writeFile, rename, unlink } from 'node:fs/promises';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { logger } from '../../utils/logger.js';\n\n/** Directory for runtime state files */\nconst RUN_DIR = join(homedir(), '.dollhouse', 'run');\n\n/** Path to the leader lock file */\nconst LOCK_FILE = join(RUN_DIR, 'console-leader.lock');\n\n/** How often the leader updates its heartbeat (ms) */\nconst HEARTBEAT_INTERVAL_MS = 10_000;\n\n/** How long before a heartbeat is considered stale (ms) */\nconst HEARTBEAT_STALE_MS = 30_000;\n\n/** Current lock file schema version */\nconst LOCK_VERSION = 1;\n\n/**\n * Information stored in the leader lock file.\n */\nexport interface ConsoleLeaderInfo {\n  version: number;\n  pid: number;\n  port: number;\n  sessionId: string;\n  startedAt: string;\n  heartbeat: string;\n}\n\n/**\n * Result of a leader election attempt.\n */\nexport interface ElectionResult {\n  role: 'leader' | 'follower';\n  /** Leader info — for followers, this is the existing leader's info */\n  leaderInfo: ConsoleLeaderInfo;\n}\n\n/**\n * Check whether a process with the given PID is alive.\n * Uses signal 0 which checks existence without sending a signal.\n */\nexport function isProcessAlive(pid: number): boolean {\n  try {\n    process.kill(pid, 0);\n    return true;\n  } catch {\n    return false;\n  }\n}\n\n/**\n * Read and parse the leader lock file.\n * Returns null if the file doesn't exist, is unreadable, or has invalid content.\n */\nexport async function readLeaderLock(): Promise<ConsoleLeaderInfo | null> {\n  try {\n    const content = await readFile(LOCK_FILE, 'utf-8');\n    const data = JSON.parse(content) as ConsoleLeaderInfo;\n    if (data.version !== LOCK_VERSION || !data.pid || !data.port || !data.sessionId) {\n      return null;\n    }\n    return data;\n  } catch {\n    return null;\n  }\n}\n\n/**\n * Check if a leader lock is stale (dead process or expired heartbeat).\n */\nexport function isLockStale(info: ConsoleLeaderInfo): boolean {\n  if (!isProcessAlive(info.pid)) {\n    return true;\n  }\n  const heartbeatAge = Date.now() - new Date(info.heartbeat).getTime();\n  return heartbeatAge > HEARTBEAT_STALE_MS;\n}\n\n/**\n * Attempt to atomically claim leadership.\n *\n * Writes to a temp file then renames to the lock path. On POSIX systems\n * rename is atomic, so only one writer wins. After renaming, re-reads the\n * lock to verify our PID won.\n *\n * @returns true if this process successfully claimed leadership\n */\nexport async function claimLeadership(info: ConsoleLeaderInfo): Promise<boolean> {\n  await mkdir(RUN_DIR, { recursive: true });\n  const tmpFile = join(RUN_DIR, `console-leader.lock.${process.pid}.tmp`);\n  try {\n    await writeFile(tmpFile, JSON.stringify(info, null, 2), 'utf-8');\n    await rename(tmpFile, LOCK_FILE);\n\n    // Verify we won the race\n    const written = await readLeaderLock();\n    return written !== null && written.pid === info.pid;\n  } catch {\n    // Clean up temp file on failure\n    try { await unlink(tmpFile); } catch { /* ignore */ }\n    return false;\n  }\n}\n\n/**\n * Delete the leader lock file (for cleanup or takeover).\n */\nexport async function deleteLeaderLock(): Promise<void> {\n  try { await unlink(LOCK_FILE); } catch { /* already gone */ }\n}\n\n/**\n * Run the leader election protocol.\n *\n * 1. If no lock exists or lock is stale → claim leadership\n * 2. If lock exists with a live, responsive leader → become follower\n *\n * @param sessionId - This process's unique session identifier\n * @param port - The port this process would use as leader (typically 3939)\n * @returns Election result with role and leader info\n */\nexport async function electLeader(sessionId: string, port: number): Promise<ElectionResult> {\n  sessionId = UnicodeValidator.normalize(sessionId).normalizedContent;\n  const existingLock = await readLeaderLock();\n\n  if (existingLock && !isLockStale(existingLock)) {\n    logger.info('[LeaderElection] Existing leader found — becoming follower', {\n      leaderSession: existingLock.sessionId, leaderPid: existingLock.pid,\n      leaderPort: existingLock.port, mySession: sessionId, myPid: process.pid,\n    });\n    return { role: 'follower', leaderInfo: existingLock };\n  }\n\n  // No valid leader — try to claim\n  if (existingLock) {\n    const alive = isProcessAlive(existingLock.pid);\n    const heartbeatAge = Date.now() - new Date(existingLock.heartbeat).getTime();\n    logger.info('[LeaderElection] Stale leader lock — taking over', {\n      stalePid: existingLock.pid, alive, heartbeatAgeMs: heartbeatAge,\n      staleSession: existingLock.sessionId, mySession: sessionId,\n    });\n    await deleteLeaderLock();\n  }\n\n  const now = new Date().toISOString();\n  const myInfo: ConsoleLeaderInfo = {\n    version: LOCK_VERSION,\n    pid: process.pid,\n    port,\n    sessionId,\n    startedAt: now,\n    heartbeat: now,\n  };\n\n  const claimed = await claimLeadership(myInfo);\n  if (claimed) {\n    logger.info('[LeaderElection] Claimed leadership', { sessionId, port, pid: process.pid });\n    return { role: 'leader', leaderInfo: myInfo };\n  }\n\n  // Another process won the race — re-read and become follower\n  const winner = await readLeaderLock();\n  if (winner) {\n    logger.info('[LeaderElection] Lost election — becoming follower', {\n      winnerPid: winner.pid, winnerSession: winner.sessionId, mySession: sessionId, myPid: process.pid,\n    });\n    return { role: 'follower', leaderInfo: winner };\n  }\n\n  // Extremely unlikely: lock disappeared between our claim and re-read. Retry once.\n  logger.warn('[LeaderElection] Lock vanished after failed claim. Retrying.');\n  const retryInfo: ConsoleLeaderInfo = { ...myInfo, heartbeat: new Date().toISOString() };\n  const retryClaimed = await claimLeadership(retryInfo);\n  if (retryClaimed) {\n    return { role: 'leader', leaderInfo: retryInfo };\n  }\n  const actualLeader = await readLeaderLock();\n  return { role: 'follower', leaderInfo: actualLeader ?? retryInfo };\n}\n\n/**\n * Start the leader heartbeat loop.\n * Updates the lock file every HEARTBEAT_INTERVAL_MS so followers know the leader is alive.\n *\n * @returns A stop function to clear the interval\n */\nexport function startHeartbeat(info: ConsoleLeaderInfo): () => void {\n  const interval = setInterval(async () => {\n    try {\n      const updated: ConsoleLeaderInfo = { ...info, heartbeat: new Date().toISOString() };\n      const tmpFile = join(RUN_DIR, `console-leader.lock.${process.pid}.tmp`);\n      await writeFile(tmpFile, JSON.stringify(updated, null, 2), 'utf-8');\n      await rename(tmpFile, LOCK_FILE);\n    } catch (err) {\n      logger.debug('[LeaderElection] Heartbeat write failed:', err);\n    }\n  }, HEARTBEAT_INTERVAL_MS);\n\n  // Don't let the heartbeat interval keep the process alive\n  interval.unref();\n\n  return () => clearInterval(interval);\n}\n\n/**\n * Register cleanup handlers to remove the leader lock on process exit.\n * Should only be called by the leader.\n */\nexport function registerLeaderCleanup(): void {\n  const cleanup = () => { deleteLeaderLock().catch(() => {}); };\n  process.once('exit', cleanup);\n  process.once('SIGTERM', cleanup);\n  process.once('SIGINT', cleanup);\n}\n"]}
248
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LeaderElection.js","sourceRoot":"","sources":["../../../src/web/console/LeaderElection.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Leader election for the unified web console.\n *\n * When multiple MCP server instances run concurrently, only one should host\n * the web console (the \"leader\"). Others become \"followers\" that forward\n * events to the leader. This module handles:\n *\n * 1. Reading/writing a leader lock file at ~/.dollhouse/run/console-leader.lock\n * 2. Atomic claim via temp+rename to prevent TOCTOU races\n * 3. PID-based stale detection (signal-0 liveness check)\n * 4. Heartbeat updates (10s interval) so followers can detect hung leaders\n * 5. Cleanup on process exit\n *\n * The port 3939 binding is the ultimate tiebreaker: even if two processes\n * both write the lock file, only one can bind the port.\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport { homedir } from 'node:os';\nimport { join } from 'node:path';\nimport { mkdir, readFile, writeFile, rename, unlink } from 'node:fs/promises';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { logger } from '../../utils/logger.js';\n\n/** Directory for runtime state files */\nconst RUN_DIR = join(homedir(), '.dollhouse', 'run');\n\n/** Path to the leader lock file */\nconst LOCK_FILE = join(RUN_DIR, 'console-leader.lock');\n\n/** How often the leader updates its heartbeat (ms) */\nconst HEARTBEAT_INTERVAL_MS = 10_000;\n\n/** How long before a heartbeat is considered stale (ms) */\nconst HEARTBEAT_STALE_MS = 30_000;\n\n/** Current lock file schema version */\nconst LOCK_VERSION = 1;\n\n/**\n * Information stored in the leader lock file.\n */\nexport interface ConsoleLeaderInfo {\n  version: number;\n  pid: number;\n  port: number;\n  sessionId: string;\n  startedAt: string;\n  heartbeat: string;\n}\n\n/**\n * Result of a leader election attempt.\n */\nexport interface ElectionResult {\n  role: 'leader' | 'follower';\n  /** Leader info — for followers, this is the existing leader's info */\n  leaderInfo: ConsoleLeaderInfo;\n}\n\n/**\n * Check whether a process with the given PID is alive.\n * Uses signal 0 which checks existence without sending a signal.\n */\nexport function isProcessAlive(pid: number): boolean {\n  try {\n    process.kill(pid, 0);\n    return true;\n  } catch {\n    return false;\n  }\n}\n\n/**\n * Read and parse the leader lock file.\n * Returns null if the file doesn't exist, is unreadable, or has invalid content.\n */\nexport async function readLeaderLock(): Promise<ConsoleLeaderInfo | null> {\n  try {\n    const content = await readFile(LOCK_FILE, 'utf-8');\n    const data = JSON.parse(content) as ConsoleLeaderInfo;\n    if (data.version !== LOCK_VERSION || !data.pid || !data.port || !data.sessionId) {\n      return null;\n    }\n    return data;\n  } catch {\n    return null;\n  }\n}\n\n/**\n * Check if a leader lock is stale (dead process or expired heartbeat).\n */\nexport function isLockStale(info: ConsoleLeaderInfo): boolean {\n  if (!isProcessAlive(info.pid)) {\n    return true;\n  }\n  const heartbeatAge = Date.now() - new Date(info.heartbeat).getTime();\n  return heartbeatAge > HEARTBEAT_STALE_MS;\n}\n\n/**\n * Attempt to atomically claim leadership.\n *\n * Writes to a temp file then renames to the lock path. On POSIX systems\n * rename is atomic, so only one writer wins. After renaming, re-reads the\n * lock to verify our PID won.\n *\n * @returns true if this process successfully claimed leadership\n */\nexport async function claimLeadership(info: ConsoleLeaderInfo): Promise<boolean> {\n  await mkdir(RUN_DIR, { recursive: true });\n  const tmpFile = join(RUN_DIR, `console-leader.lock.${process.pid}.tmp`);\n  try {\n    await writeFile(tmpFile, JSON.stringify(info, null, 2), 'utf-8');\n    await rename(tmpFile, LOCK_FILE);\n\n    // Verify we won the race\n    const written = await readLeaderLock();\n    return written !== null && written.pid === info.pid;\n  } catch {\n    // Clean up temp file on failure\n    try { await unlink(tmpFile); } catch { /* ignore */ }\n    return false;\n  }\n}\n\n/**\n * Delete the leader lock file (for cleanup or takeover).\n */\nexport async function deleteLeaderLock(): Promise<void> {\n  try { await unlink(LOCK_FILE); } catch { /* already gone */ }\n}\n\n/**\n * Run the leader election protocol.\n *\n * 1. If no lock exists or lock is stale → claim leadership\n * 2. If lock exists with a live, responsive leader → become follower\n *\n * @param sessionId - This process's unique session identifier\n * @param port - The port this process would use as leader (typically 3939)\n * @returns Election result with role and leader info\n */\nexport async function electLeader(sessionId: string, port: number): Promise<ElectionResult> {\n  sessionId = UnicodeValidator.normalize(sessionId).normalizedContent;\n  const existingLock = await readLeaderLock();\n\n  if (existingLock && !isLockStale(existingLock)) {\n    logger.info('[LeaderElection] Existing leader found — becoming follower', {\n      leaderSession: existingLock.sessionId, leaderPid: existingLock.pid,\n      leaderPort: existingLock.port, mySession: sessionId, myPid: process.pid,\n    });\n    return { role: 'follower', leaderInfo: existingLock };\n  }\n\n  // No valid leader — try to claim\n  if (existingLock) {\n    const alive = isProcessAlive(existingLock.pid);\n    const heartbeatAge = Date.now() - new Date(existingLock.heartbeat).getTime();\n    logger.info('[LeaderElection] Stale leader lock — taking over', {\n      stalePid: existingLock.pid, alive, heartbeatAgeMs: heartbeatAge,\n      staleSession: existingLock.sessionId, mySession: sessionId,\n    });\n    await deleteLeaderLock();\n  }\n\n  const now = new Date().toISOString();\n  const myInfo: ConsoleLeaderInfo = {\n    version: LOCK_VERSION,\n    pid: process.pid,\n    port,\n    sessionId,\n    startedAt: now,\n    heartbeat: now,\n  };\n\n  const claimed = await claimLeadership(myInfo);\n  if (claimed) {\n    logger.info('[LeaderElection] Claimed leadership', { sessionId, port, pid: process.pid });\n    return { role: 'leader', leaderInfo: myInfo };\n  }\n\n  // Another process won the race — re-read and become follower\n  const winner = await readLeaderLock();\n  if (winner) {\n    logger.info('[LeaderElection] Lost election — becoming follower', {\n      winnerPid: winner.pid, winnerSession: winner.sessionId, mySession: sessionId, myPid: process.pid,\n    });\n    return { role: 'follower', leaderInfo: winner };\n  }\n\n  // Extremely unlikely: lock disappeared between our claim and re-read. Retry once.\n  logger.warn('[LeaderElection] Lock vanished after failed claim. Retrying.');\n  const retryInfo: ConsoleLeaderInfo = { ...myInfo, heartbeat: new Date().toISOString() };\n  const retryClaimed = await claimLeadership(retryInfo);\n  if (retryClaimed) {\n    return { role: 'leader', leaderInfo: retryInfo };\n  }\n  const actualLeader = await readLeaderLock();\n  return { role: 'follower', leaderInfo: actualLeader ?? retryInfo };\n}\n\n/**\n * Probe whether the leader's web console is reachable.\n * Returns true if the leader's ingest endpoint responds, false otherwise.\n */\nexport async function isLeaderWebConsoleReachable(leaderInfo: ConsoleLeaderInfo): Promise<boolean> {\n  try {\n    const controller = new AbortController();\n    const timeout = setTimeout(() => controller.abort(), 2_000);\n    const res = await fetch(`http://127.0.0.1:${leaderInfo.port}/api/logs/stats`, {\n      signal: controller.signal,\n    });\n    clearTimeout(timeout);\n    return res.ok;\n  } catch {\n    return false;\n  }\n}\n\n/**\n * Force claim leadership by deleting the existing lock and claiming.\n * Used when the existing leader is alive but not running a web console.\n */\nexport async function forceClaimLeadership(sessionId: string, port: number): Promise<ElectionResult> {\n  logger.info('[LeaderElection] Forcing leadership takeover — existing leader has no web console');\n  await deleteLeaderLock();\n\n  const now = new Date().toISOString();\n  const myInfo: ConsoleLeaderInfo = {\n    version: LOCK_VERSION,\n    pid: process.pid,\n    port,\n    sessionId: UnicodeValidator.normalize(sessionId).normalizedContent,\n    startedAt: now,\n    heartbeat: now,\n  };\n\n  const claimed = await claimLeadership(myInfo);\n  if (claimed) {\n    logger.info('[LeaderElection] Forced leadership claimed', { sessionId, port, pid: process.pid });\n    return { role: 'leader', leaderInfo: myInfo };\n  }\n\n  // Failed — fall back to follower\n  const winner = await readLeaderLock();\n  return { role: 'follower', leaderInfo: winner ?? myInfo };\n}\n\n/**\n * Start the leader heartbeat loop.\n * Updates the lock file every HEARTBEAT_INTERVAL_MS so followers know the leader is alive.\n *\n * @returns A stop function to clear the interval\n */\nexport function startHeartbeat(info: ConsoleLeaderInfo): () => void {\n  const interval = setInterval(async () => {\n    try {\n      const updated: ConsoleLeaderInfo = { ...info, heartbeat: new Date().toISOString() };\n      const tmpFile = join(RUN_DIR, `console-leader.lock.${process.pid}.tmp`);\n      await writeFile(tmpFile, JSON.stringify(updated, null, 2), 'utf-8');\n      await rename(tmpFile, LOCK_FILE);\n    } catch (err) {\n      logger.debug('[LeaderElection] Heartbeat write failed:', err);\n    }\n  }, HEARTBEAT_INTERVAL_MS);\n\n  // Don't let the heartbeat interval keep the process alive\n  interval.unref();\n\n  return () => clearInterval(interval);\n}\n\n/**\n * Register cleanup handlers to remove the leader lock on process exit.\n * Should only be called by the leader.\n */\nexport function registerLeaderCleanup(): void {\n  const cleanup = () => { deleteLeaderLock().catch(() => {}); };\n  process.once('exit', cleanup);\n  process.once('SIGTERM', cleanup);\n  process.once('SIGINT', cleanup);\n}\n"]}
@@ -26,13 +26,15 @@ export declare class LeaderForwardingLogSink implements ILogSink {
26
26
  private flushTimer;
27
27
  private backoffMs;
28
28
  private flushing;
29
+ private consecutiveFailures;
30
+ private gaveUp;
29
31
  constructor(leaderUrl: string, sessionId: string);
30
32
  write(entry: UnifiedLogEntry): void;
31
33
  flush(): Promise<void>;
32
34
  close(): Promise<void>;
33
35
  private flushBuffer;
34
36
  private requeueBatch;
35
- private scheduleRetry;
37
+ private handleFailure;
36
38
  }
37
39
  /**
38
40
  * Forwards metric snapshots to the leader's /api/ingest/metrics.
@@ -1 +1 @@
1
- {"version":3,"file":"LeaderForwardingSink.d.ts","sourceRoot":"","sources":["../../../src/web/console/LeaderForwardingSink.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAO,KAAK,EAAE,QAAQ,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACxE,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AAsB7D;;GAEG;AACH,qBAAa,uBAAwB,YAAW,QAAQ;IAOpD,OAAO,CAAC,QAAQ,CAAC,SAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,SAAS;IAP5B,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAyB;IAChD,OAAO,CAAC,UAAU,CAA+C;IACjE,OAAO,CAAC,SAAS,CAAsB;IACvC,OAAO,CAAC,QAAQ,CAAS;gBAGN,SAAS,EAAE,MAAM,EACjB,SAAS,EAAE,MAAM;IAOpC,KAAK,CAAC,KAAK,EAAE,eAAe,GAAG,IAAI;IAkB7B,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAItB,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;YAQd,WAAW;IA+BzB,OAAO,CAAC,YAAY;IAUpB,OAAO,CAAC,aAAa;CAItB;AAED;;GAEG;AACH,qBAAa,2BAA2B;IAEpC,OAAO,CAAC,QAAQ,CAAC,SAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,SAAS;gBADT,SAAS,EAAE,MAAM,EACjB,SAAS,EAAE,MAAM;IAG9B,UAAU,CAAC,QAAQ,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC;CAgB1D;AAED;;GAEG;AACH,qBAAa,gBAAgB;IAIzB,OAAO,CAAC,QAAQ,CAAC,SAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,SAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,GAAG;IALtB,OAAO,CAAC,cAAc,CAA+C;gBAGlD,SAAS,EAAE,MAAM,EACjB,SAAS,EAAE,MAAM,EACjB,GAAG,EAAE,MAAM;IAG9B,sDAAsD;IAChD,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAS5B,sDAAsD;IAChD,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;YAQb,SAAS;CAqBxB"}
1
+ {"version":3,"file":"LeaderForwardingSink.d.ts","sourceRoot":"","sources":["../../../src/web/console/LeaderForwardingSink.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAO,KAAK,EAAE,QAAQ,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACxE,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AAyB7D;;GAEG;AACH,qBAAa,uBAAwB,YAAW,QAAQ;IASpD,OAAO,CAAC,QAAQ,CAAC,SAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,SAAS;IAT5B,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAyB;IAChD,OAAO,CAAC,UAAU,CAA+C;IACjE,OAAO,CAAC,SAAS,CAAsB;IACvC,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,mBAAmB,CAAK;IAChC,OAAO,CAAC,MAAM,CAAS;gBAGJ,SAAS,EAAE,MAAM,EACjB,SAAS,EAAE,MAAM;IAOpC,KAAK,CAAC,KAAK,EAAE,eAAe,GAAG,IAAI;IAkB7B,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAItB,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;YAQd,WAAW;IAgCzB,OAAO,CAAC,YAAY;IAUpB,OAAO,CAAC,aAAa;CAmBtB;AAED;;GAEG;AACH,qBAAa,2BAA2B;IAEpC,OAAO,CAAC,QAAQ,CAAC,SAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,SAAS;gBADT,SAAS,EAAE,MAAM,EACjB,SAAS,EAAE,MAAM;IAG9B,UAAU,CAAC,QAAQ,EAAE,cAAc,GAAG,OAAO,CAAC,IAAI,CAAC;CAgB1D;AAED;;GAEG;AACH,qBAAa,gBAAgB;IAIzB,OAAO,CAAC,QAAQ,CAAC,SAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,SAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,GAAG;IALtB,OAAO,CAAC,cAAc,CAA+C;gBAGlD,SAAS,EAAE,MAAM,EACjB,SAAS,EAAE,MAAM,EACjB,GAAG,EAAE,MAAM;IAG9B,sDAAsD;IAChD,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAS5B,sDAAsD;IAChD,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;YAQb,SAAS;CAqBxB"}
@@ -26,6 +26,8 @@ const FLUSH_INTERVAL_MS = 1_000;
26
26
  const INITIAL_BACKOFF_MS = 1_000;
27
27
  /** Maximum backoff delay (ms) */
28
28
  const MAX_BACKOFF_MS = 30_000;
29
+ /** Give up forwarding after this many consecutive failures */
30
+ const MAX_CONSECUTIVE_FAILURES = 5;
29
31
  /** HTTP request timeout (ms) */
30
32
  const REQUEST_TIMEOUT_MS = 5_000;
31
33
  /**
@@ -38,6 +40,8 @@ export class LeaderForwardingLogSink {
38
40
  flushTimer = null;
39
41
  backoffMs = INITIAL_BACKOFF_MS;
40
42
  flushing = false;
43
+ consecutiveFailures = 0;
44
+ gaveUp = false;
41
45
  constructor(leaderUrl, sessionId) {
42
46
  this.leaderUrl = leaderUrl;
43
47
  this.sessionId = sessionId;
@@ -71,7 +75,7 @@ export class LeaderForwardingLogSink {
71
75
  await this.flushBuffer();
72
76
  }
73
77
  async flushBuffer() {
74
- if (this.flushing || this.buffer.length === 0)
78
+ if (this.flushing || this.buffer.length === 0 || this.gaveUp)
75
79
  return;
76
80
  this.flushing = true;
77
81
  const batch = this.buffer.splice(0, BATCH_SIZE);
@@ -87,15 +91,16 @@ export class LeaderForwardingLogSink {
87
91
  clearTimeout(timeout);
88
92
  if (response.ok) {
89
93
  this.backoffMs = INITIAL_BACKOFF_MS;
94
+ this.consecutiveFailures = 0;
90
95
  }
91
96
  else {
92
97
  this.requeueBatch(batch);
93
- this.scheduleRetry();
98
+ this.handleFailure();
94
99
  }
95
100
  }
96
101
  catch {
97
102
  this.requeueBatch(batch);
98
- this.scheduleRetry();
103
+ this.handleFailure();
99
104
  }
100
105
  finally {
101
106
  this.flushing = false;
@@ -111,8 +116,21 @@ export class LeaderForwardingLogSink {
111
116
  logger.warn(`[ForwardingSink] Buffer full (${MAX_BUFFER_SIZE}), dropping ${batch.length} entries`);
112
117
  }
113
118
  }
114
- scheduleRetry() {
115
- logger.debug(`[ForwardingSink] Leader unreachable, backoff ${this.backoffMs}ms (buffered: ${this.buffer.length})`);
119
+ handleFailure() {
120
+ this.consecutiveFailures++;
121
+ if (this.consecutiveFailures >= MAX_CONSECUTIVE_FAILURES) {
122
+ if (!this.gaveUp) {
123
+ this.gaveUp = true;
124
+ logger.info(`[ForwardingSink] Leader not running web console — log forwarding disabled after ${this.consecutiveFailures} failed attempts. Buffered ${this.buffer.length} entries discarded.`);
125
+ this.buffer.length = 0;
126
+ if (this.flushTimer) {
127
+ clearInterval(this.flushTimer);
128
+ this.flushTimer = null;
129
+ }
130
+ }
131
+ return;
132
+ }
133
+ logger.debug(`[ForwardingSink] Leader unreachable, backoff ${this.backoffMs}ms (attempt ${this.consecutiveFailures}/${MAX_CONSECUTIVE_FAILURES}, buffered: ${this.buffer.length})`);
116
134
  this.backoffMs = Math.min(this.backoffMs * 2, MAX_BACKOFF_MS);
117
135
  }
118
136
  }
@@ -194,4 +212,4 @@ export class SessionHeartbeat {
194
212
  }
195
213
  }
196
214
  }
197
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LeaderForwardingSink.js","sourceRoot":"","sources":["../../../src/web/console/LeaderForwardingSink.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Forwarding sinks for follower MCP servers.\n *\n * When a server becomes a follower in the unified console election, it\n * registers these sinks with its LogManager and MetricsManager. Instead\n * of broadcasting to local SSE clients, entries are batch-POSTed to\n * the leader's ingestion endpoints.\n *\n * Features:\n * - Batch buffering (50 entries or 1s flush, whichever comes first)\n * - In-memory buffer up to 10,000 entries on leader failure\n * - Exponential backoff on POST failure (1s → 2s → 4s, max 30s)\n * - Automatic drain on leader recovery\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport type { ILogSink, UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { logger } from '../../utils/logger.js';\n\n/** Maximum entries to buffer when leader is unreachable */\nconst MAX_BUFFER_SIZE = 10_000;\n\n/** Batch size before flushing */\nconst BATCH_SIZE = 50;\n\n/** Time-based flush interval (ms) */\nconst FLUSH_INTERVAL_MS = 1_000;\n\n/** Initial backoff delay on failure (ms) */\nconst INITIAL_BACKOFF_MS = 1_000;\n\n/** Maximum backoff delay (ms) */\nconst MAX_BACKOFF_MS = 30_000;\n\n/** HTTP request timeout (ms) */\nconst REQUEST_TIMEOUT_MS = 5_000;\n\n/**\n * ILogSink that batch-POSTs entries to the leader's /api/ingest/logs.\n */\nexport class LeaderForwardingLogSink implements ILogSink {\n  private readonly buffer: UnifiedLogEntry[] = [];\n  private flushTimer: ReturnType<typeof setInterval> | null = null;\n  private backoffMs = INITIAL_BACKOFF_MS;\n  private flushing = false;\n\n  constructor(\n    private readonly leaderUrl: string,\n    private readonly sessionId: string,\n  ) {\n    this.sessionId = UnicodeValidator.normalize(sessionId).normalizedContent;\n    this.flushTimer = setInterval(() => this.flushBuffer(), FLUSH_INTERVAL_MS);\n    this.flushTimer.unref();\n  }\n\n  write(entry: UnifiedLogEntry): void {\n    // Stamp session ID before buffering\n    const stamped: UnifiedLogEntry = {\n      ...entry,\n      data: { ...entry.data, _sessionId: this.sessionId },\n    };\n\n    if (this.buffer.length >= MAX_BUFFER_SIZE) {\n      // Evict oldest entry (FIFO)\n      this.buffer.shift();\n    }\n    this.buffer.push(stamped);\n\n    if (this.buffer.length >= BATCH_SIZE) {\n      this.flushBuffer();\n    }\n  }\n\n  async flush(): Promise<void> {\n    await this.flushBuffer();\n  }\n\n  async close(): Promise<void> {\n    if (this.flushTimer) {\n      clearInterval(this.flushTimer);\n      this.flushTimer = null;\n    }\n    await this.flushBuffer();\n  }\n\n  private async flushBuffer(): Promise<void> {\n    if (this.flushing || this.buffer.length === 0) return;\n    this.flushing = true;\n\n    const batch = this.buffer.splice(0, BATCH_SIZE);\n    try {\n      const controller = new AbortController();\n      const timeout = setTimeout(() => controller.abort(), REQUEST_TIMEOUT_MS);\n\n      const response = await fetch(`${this.leaderUrl}/api/ingest/logs`, {\n        method: 'POST',\n        headers: { 'Content-Type': 'application/json' },\n        body: JSON.stringify({ sessionId: this.sessionId, entries: batch }),\n        signal: controller.signal,\n      });\n      clearTimeout(timeout);\n\n      if (response.ok) {\n        this.backoffMs = INITIAL_BACKOFF_MS;\n      } else {\n        this.requeueBatch(batch);\n        this.scheduleRetry();\n      }\n    } catch {\n      this.requeueBatch(batch);\n      this.scheduleRetry();\n    } finally {\n      this.flushing = false;\n    }\n  }\n\n  private requeueBatch(batch: UnifiedLogEntry[]): void {\n    const spaceAvailable = MAX_BUFFER_SIZE - this.buffer.length;\n    if (spaceAvailable > 0) {\n      const toRequeue = batch.slice(0, spaceAvailable);\n      this.buffer.unshift(...toRequeue);\n    } else {\n      logger.warn(`[ForwardingSink] Buffer full (${MAX_BUFFER_SIZE}), dropping ${batch.length} entries`);\n    }\n  }\n\n  private scheduleRetry(): void {\n    logger.debug(`[ForwardingSink] Leader unreachable, backoff ${this.backoffMs}ms (buffered: ${this.buffer.length})`);\n    this.backoffMs = Math.min(this.backoffMs * 2, MAX_BACKOFF_MS);\n  }\n}\n\n/**\n * Forwards metric snapshots to the leader's /api/ingest/metrics.\n */\nexport class LeaderForwardingMetricsSink {\n  constructor(\n    private readonly leaderUrl: string,\n    private readonly sessionId: string,\n  ) {}\n\n  async onSnapshot(snapshot: MetricSnapshot): Promise<void> {\n    try {\n      const controller = new AbortController();\n      const timeout = setTimeout(() => controller.abort(), REQUEST_TIMEOUT_MS);\n\n      await fetch(`${this.leaderUrl}/api/ingest/metrics`, {\n        method: 'POST',\n        headers: { 'Content-Type': 'application/json' },\n        body: JSON.stringify({ sessionId: this.sessionId, snapshot }),\n        signal: controller.signal,\n      });\n      clearTimeout(timeout);\n    } catch {\n      logger.debug('[ForwardingSink] Failed to forward metrics snapshot');\n    }\n  }\n}\n\n/**\n * Sends session lifecycle events to the leader.\n */\nexport class SessionHeartbeat {\n  private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n\n  constructor(\n    private readonly leaderUrl: string,\n    private readonly sessionId: string,\n    private readonly pid: number,\n  ) {}\n\n  /** Notify the leader that this session has started */\n  async start(): Promise<void> {\n    await this.sendEvent('started');\n\n    this.heartbeatTimer = setInterval(() => {\n      this.sendEvent('heartbeat').catch(() => {});\n    }, 10_000);\n    this.heartbeatTimer.unref();\n  }\n\n  /** Notify the leader that this session is stopping */\n  async stop(): Promise<void> {\n    if (this.heartbeatTimer) {\n      clearInterval(this.heartbeatTimer);\n      this.heartbeatTimer = null;\n    }\n    await this.sendEvent('stopped');\n  }\n\n  private async sendEvent(event: 'started' | 'stopped' | 'heartbeat'): Promise<void> {\n    try {\n      const controller = new AbortController();\n      const timeout = setTimeout(() => controller.abort(), REQUEST_TIMEOUT_MS);\n\n      await fetch(`${this.leaderUrl}/api/ingest/session`, {\n        method: 'POST',\n        headers: { 'Content-Type': 'application/json' },\n        body: JSON.stringify({\n          sessionId: this.sessionId,\n          event,\n          pid: this.pid,\n          startedAt: new Date().toISOString(),\n        }),\n        signal: controller.signal,\n      });\n      clearTimeout(timeout);\n    } catch {\n      logger.debug(`[SessionHeartbeat] Failed to send ${event} event`);\n    }\n  }\n}\n"]}
215
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LeaderForwardingSink.js","sourceRoot":"","sources":["../../../src/web/console/LeaderForwardingSink.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Forwarding sinks for follower MCP servers.\n *\n * When a server becomes a follower in the unified console election, it\n * registers these sinks with its LogManager and MetricsManager. Instead\n * of broadcasting to local SSE clients, entries are batch-POSTed to\n * the leader's ingestion endpoints.\n *\n * Features:\n * - Batch buffering (50 entries or 1s flush, whichever comes first)\n * - In-memory buffer up to 10,000 entries on leader failure\n * - Exponential backoff on POST failure (1s → 2s → 4s, max 30s)\n * - Automatic drain on leader recovery\n *\n * @since v2.1.0 — Issue #1700\n */\n\nimport type { ILogSink, UnifiedLogEntry } from '../../logging/types.js';\nimport type { MetricSnapshot } from '../../metrics/types.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { logger } from '../../utils/logger.js';\n\n/** Maximum entries to buffer when leader is unreachable */\nconst MAX_BUFFER_SIZE = 10_000;\n\n/** Batch size before flushing */\nconst BATCH_SIZE = 50;\n\n/** Time-based flush interval (ms) */\nconst FLUSH_INTERVAL_MS = 1_000;\n\n/** Initial backoff delay on failure (ms) */\nconst INITIAL_BACKOFF_MS = 1_000;\n\n/** Maximum backoff delay (ms) */\nconst MAX_BACKOFF_MS = 30_000;\n\n/** Give up forwarding after this many consecutive failures */\nconst MAX_CONSECUTIVE_FAILURES = 5;\n\n/** HTTP request timeout (ms) */\nconst REQUEST_TIMEOUT_MS = 5_000;\n\n/**\n * ILogSink that batch-POSTs entries to the leader's /api/ingest/logs.\n */\nexport class LeaderForwardingLogSink implements ILogSink {\n  private readonly buffer: UnifiedLogEntry[] = [];\n  private flushTimer: ReturnType<typeof setInterval> | null = null;\n  private backoffMs = INITIAL_BACKOFF_MS;\n  private flushing = false;\n  private consecutiveFailures = 0;\n  private gaveUp = false;\n\n  constructor(\n    private readonly leaderUrl: string,\n    private readonly sessionId: string,\n  ) {\n    this.sessionId = UnicodeValidator.normalize(sessionId).normalizedContent;\n    this.flushTimer = setInterval(() => this.flushBuffer(), FLUSH_INTERVAL_MS);\n    this.flushTimer.unref();\n  }\n\n  write(entry: UnifiedLogEntry): void {\n    // Stamp session ID before buffering\n    const stamped: UnifiedLogEntry = {\n      ...entry,\n      data: { ...entry.data, _sessionId: this.sessionId },\n    };\n\n    if (this.buffer.length >= MAX_BUFFER_SIZE) {\n      // Evict oldest entry (FIFO)\n      this.buffer.shift();\n    }\n    this.buffer.push(stamped);\n\n    if (this.buffer.length >= BATCH_SIZE) {\n      this.flushBuffer();\n    }\n  }\n\n  async flush(): Promise<void> {\n    await this.flushBuffer();\n  }\n\n  async close(): Promise<void> {\n    if (this.flushTimer) {\n      clearInterval(this.flushTimer);\n      this.flushTimer = null;\n    }\n    await this.flushBuffer();\n  }\n\n  private async flushBuffer(): Promise<void> {\n    if (this.flushing || this.buffer.length === 0 || this.gaveUp) return;\n    this.flushing = true;\n\n    const batch = this.buffer.splice(0, BATCH_SIZE);\n    try {\n      const controller = new AbortController();\n      const timeout = setTimeout(() => controller.abort(), REQUEST_TIMEOUT_MS);\n\n      const response = await fetch(`${this.leaderUrl}/api/ingest/logs`, {\n        method: 'POST',\n        headers: { 'Content-Type': 'application/json' },\n        body: JSON.stringify({ sessionId: this.sessionId, entries: batch }),\n        signal: controller.signal,\n      });\n      clearTimeout(timeout);\n\n      if (response.ok) {\n        this.backoffMs = INITIAL_BACKOFF_MS;\n        this.consecutiveFailures = 0;\n      } else {\n        this.requeueBatch(batch);\n        this.handleFailure();\n      }\n    } catch {\n      this.requeueBatch(batch);\n      this.handleFailure();\n    } finally {\n      this.flushing = false;\n    }\n  }\n\n  private requeueBatch(batch: UnifiedLogEntry[]): void {\n    const spaceAvailable = MAX_BUFFER_SIZE - this.buffer.length;\n    if (spaceAvailable > 0) {\n      const toRequeue = batch.slice(0, spaceAvailable);\n      this.buffer.unshift(...toRequeue);\n    } else {\n      logger.warn(`[ForwardingSink] Buffer full (${MAX_BUFFER_SIZE}), dropping ${batch.length} entries`);\n    }\n  }\n\n  private handleFailure(): void {\n    this.consecutiveFailures++;\n\n    if (this.consecutiveFailures >= MAX_CONSECUTIVE_FAILURES) {\n      if (!this.gaveUp) {\n        this.gaveUp = true;\n        logger.info(`[ForwardingSink] Leader not running web console — log forwarding disabled after ${this.consecutiveFailures} failed attempts. Buffered ${this.buffer.length} entries discarded.`);\n        this.buffer.length = 0;\n        if (this.flushTimer) {\n          clearInterval(this.flushTimer);\n          this.flushTimer = null;\n        }\n      }\n      return;\n    }\n\n    logger.debug(`[ForwardingSink] Leader unreachable, backoff ${this.backoffMs}ms (attempt ${this.consecutiveFailures}/${MAX_CONSECUTIVE_FAILURES}, buffered: ${this.buffer.length})`);\n    this.backoffMs = Math.min(this.backoffMs * 2, MAX_BACKOFF_MS);\n  }\n}\n\n/**\n * Forwards metric snapshots to the leader's /api/ingest/metrics.\n */\nexport class LeaderForwardingMetricsSink {\n  constructor(\n    private readonly leaderUrl: string,\n    private readonly sessionId: string,\n  ) {}\n\n  async onSnapshot(snapshot: MetricSnapshot): Promise<void> {\n    try {\n      const controller = new AbortController();\n      const timeout = setTimeout(() => controller.abort(), REQUEST_TIMEOUT_MS);\n\n      await fetch(`${this.leaderUrl}/api/ingest/metrics`, {\n        method: 'POST',\n        headers: { 'Content-Type': 'application/json' },\n        body: JSON.stringify({ sessionId: this.sessionId, snapshot }),\n        signal: controller.signal,\n      });\n      clearTimeout(timeout);\n    } catch {\n      logger.debug('[ForwardingSink] Failed to forward metrics snapshot');\n    }\n  }\n}\n\n/**\n * Sends session lifecycle events to the leader.\n */\nexport class SessionHeartbeat {\n  private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n\n  constructor(\n    private readonly leaderUrl: string,\n    private readonly sessionId: string,\n    private readonly pid: number,\n  ) {}\n\n  /** Notify the leader that this session has started */\n  async start(): Promise<void> {\n    await this.sendEvent('started');\n\n    this.heartbeatTimer = setInterval(() => {\n      this.sendEvent('heartbeat').catch(() => {});\n    }, 10_000);\n    this.heartbeatTimer.unref();\n  }\n\n  /** Notify the leader that this session is stopping */\n  async stop(): Promise<void> {\n    if (this.heartbeatTimer) {\n      clearInterval(this.heartbeatTimer);\n      this.heartbeatTimer = null;\n    }\n    await this.sendEvent('stopped');\n  }\n\n  private async sendEvent(event: 'started' | 'stopped' | 'heartbeat'): Promise<void> {\n    try {\n      const controller = new AbortController();\n      const timeout = setTimeout(() => controller.abort(), REQUEST_TIMEOUT_MS);\n\n      await fetch(`${this.leaderUrl}/api/ingest/session`, {\n        method: 'POST',\n        headers: { 'Content-Type': 'application/json' },\n        body: JSON.stringify({\n          sessionId: this.sessionId,\n          event,\n          pid: this.pid,\n          startedAt: new Date().toISOString(),\n        }),\n        signal: controller.signal,\n      });\n      clearTimeout(timeout);\n    } catch {\n      logger.debug(`[SessionHeartbeat] Failed to send ${event} event`);\n    }\n  }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"UnifiedConsole.d.ts","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAEH,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAC9D,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AAC7D,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,0CAA0C,CAAC;AAElF,OAAO,EAIL,KAAK,cAAc,EACpB,MAAM,qBAAqB,CAAC;AAU7B;;GAEG;AACH,MAAM,WAAW,qBAAqB;IACpC,uCAAuC;IACvC,SAAS,EAAE,MAAM,CAAC;IAClB,oDAAoD;IACpD,YAAY,EAAE,MAAM,CAAC;IACrB,4CAA4C;IAC5C,UAAU,EAAE,aAAa,CAAC;IAC1B,0BAA0B;IAC1B,WAAW,CAAC,EAAE,iBAAiB,CAAC;IAChC,qFAAqF;IACrF,aAAa,CAAC,EAAE,GAAG,CAAC;IACpB,0DAA0D;IAC1D,eAAe,EAAE,CAAC,IAAI,EAAE;QAAE,KAAK,CAAC,KAAK,EAAE,eAAe,GAAG,IAAI,CAAC;QAAC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC;QAAC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC,CAAA;KAAE,KAAK,IAAI,CAAC;IACzH,8DAA8D;IAC9D,iBAAiB,EAAE,CAAC,SAAS,EAAE;QAAE,YAAY,CAAC,EAAE,CAAC,KAAK,EAAE,eAAe,KAAK,IAAI,CAAC;QAAC,iBAAiB,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,KAAK,IAAI,CAAA;KAAE,EAAE,WAAW,CAAC,EAAE,iBAAiB,KAAK,IAAI,CAAC;CACtL;AAED;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC,IAAI,EAAE,QAAQ,GAAG,UAAU,CAAC;IAC5B,QAAQ,EAAE,cAAc,CAAC;IACzB,mDAAmD;IACnD,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,2CAA2C;IAC3C,OAAO,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;CAC9B;AAED;;;;;GAKG;AACH,wBAAsB,mBAAmB,CAAC,OAAO,EAAE,qBAAqB,GAAG,OAAO,CAAC,oBAAoB,CAAC,CAQvG"}
1
+ {"version":3,"file":"UnifiedConsole.d.ts","sourceRoot":"","sources":["../../../src/web/console/UnifiedConsole.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAEH,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AAC9D,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,wBAAwB,CAAC;AAC7D,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,0CAA0C,CAAC;AAElF,OAAO,EAML,KAAK,cAAc,EACpB,MAAM,qBAAqB,CAAC;AAU7B;;GAEG;AACH,MAAM,WAAW,qBAAqB;IACpC,uCAAuC;IACvC,SAAS,EAAE,MAAM,CAAC;IAClB,oDAAoD;IACpD,YAAY,EAAE,MAAM,CAAC;IACrB,4CAA4C;IAC5C,UAAU,EAAE,aAAa,CAAC;IAC1B,0BAA0B;IAC1B,WAAW,CAAC,EAAE,iBAAiB,CAAC;IAChC,qFAAqF;IACrF,aAAa,CAAC,EAAE,GAAG,CAAC;IACpB,0DAA0D;IAC1D,eAAe,EAAE,CAAC,IAAI,EAAE;QAAE,KAAK,CAAC,KAAK,EAAE,eAAe,GAAG,IAAI,CAAC;QAAC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC,CAAC;QAAC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC,CAAA;KAAE,KAAK,IAAI,CAAC;IACzH,8DAA8D;IAC9D,iBAAiB,EAAE,CAAC,SAAS,EAAE;QAAE,YAAY,CAAC,EAAE,CAAC,KAAK,EAAE,eAAe,KAAK,IAAI,CAAC;QAAC,iBAAiB,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,KAAK,IAAI,CAAA;KAAE,EAAE,WAAW,CAAC,EAAE,iBAAiB,KAAK,IAAI,CAAC;CACtL;AAED;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC,IAAI,EAAE,QAAQ,GAAG,UAAU,CAAC;IAC5B,QAAQ,EAAE,cAAc,CAAC;IACzB,mDAAmD;IACnD,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,2CAA2C;IAC3C,OAAO,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;CAC9B;AAED;;;;;GAKG;AACH,wBAAsB,mBAAmB,CAAC,OAAO,EAAE,qBAAqB,GAAG,OAAO,CAAC,oBAAoB,CAAC,CAkBvG"}