@dollhousemcp/mcp-server 2.0.5 → 2.0.7

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.
@@ -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,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"}
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;IAG/B,OAAO,CAAC,QAAQ,CAEQ;IACxB,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,7 +13,10 @@ import { EvictingQueue } from './EvictingQueue.js';
13
13
  class MCPLogger {
14
14
  logs = new EvictingQueue(1000);
15
15
  isMCPConnected = false;
16
- minLevel = 'debug';
16
+ // In --web mode, default to error-only so bootstrap logs don't flood the terminal.
17
+ // All levels still go to MemoryLogSink (visible in the Logs tab).
18
+ minLevel = (process.argv.includes('--web') && !process.env.DOLLHOUSE_DEBUG && !process.env.ENABLE_DEBUG)
19
+ ? 'error' : 'debug';
17
20
  static LEVEL_ORDER = { debug: 0, info: 1, warn: 2, error: 3 };
18
21
  logListener;
19
22
  addLogListener(fn) {
@@ -296,4 +299,4 @@ class MCPLogger {
296
299
  export { MCPLogger };
297
300
  // Singleton instance for convenience
298
301
  export 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();"]}
302
+ //# 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  // In --web mode, default to error-only so bootstrap logs don't flood the terminal.\n  // All levels still go to MemoryLogSink (visible in the Logs tab).\n  private minLevel: 'debug' | 'info' | 'warn' | 'error' =\n    (process.argv.includes('--web') && !process.env.DOLLHOUSE_DEBUG && !process.env.ENABLE_DEBUG)\n      ? '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();"]}
@@ -1941,14 +1941,32 @@ function safeParseYaml(content) {
1941
1941
  });
1942
1942
  };
1943
1943
 
1944
- const savedTab = localStorage.getItem(TAB_KEY);
1945
- if (savedTab) {
1946
- switchToTab(savedTab);
1947
- } else if (!localStorage.getItem(SETUP_SEEN_KEY)) {
1948
- localStorage.setItem(SETUP_SEEN_KEY, '1');
1949
- switchToTab('setup');
1944
+ // Tab selection priority: URL hash > localStorage > first-visit setup > portfolio default
1945
+ function applyHashTab() {
1946
+ const hashTab = globalThis.location.hash.replace('#', '');
1947
+ if (hashTab && document.getElementById('tab-' + hashTab)) {
1948
+ switchToTab(hashTab);
1949
+ lazyInitTab(hashTab, tabInits);
1950
+ localStorage.setItem(TAB_KEY, hashTab);
1951
+ history.replaceState(null, '', globalThis.location.pathname);
1952
+ return true;
1953
+ }
1954
+ return false;
1950
1955
  }
1951
1956
 
1957
+ if (!applyHashTab()) {
1958
+ const savedTab = localStorage.getItem(TAB_KEY);
1959
+ if (savedTab) {
1960
+ switchToTab(savedTab);
1961
+ } else if (!localStorage.getItem(SETUP_SEEN_KEY)) {
1962
+ localStorage.setItem(SETUP_SEEN_KEY, '1');
1963
+ switchToTab('setup');
1964
+ }
1965
+ }
1966
+
1967
+ // Handle hash changes for deep-linking (e.g., open_logs operation)
1968
+ globalThis.addEventListener('hashchange', () => applyHashTab());
1969
+
1952
1970
  if (consoleTabs) {
1953
1971
  consoleTabs.addEventListener('click', (e) => {
1954
1972
  const btn = e.target.closest('.console-tab');
@@ -13,6 +13,8 @@
13
13
  import type { MCPAQLHandler } from '../handlers/mcp-aql/MCPAQLHandler.js';
14
14
  import type { MemoryLogSink } from '../logging/sinks/MemoryLogSink.js';
15
15
  import type { MemoryMetricsSink } from '../metrics/sinks/MemoryMetricsSink.js';
16
+ /** Check whether the web server has been started in this process. */
17
+ export declare function isWebServerRunning(): boolean;
16
18
  /**
17
19
  * Options for starting the web server.
18
20
  */
@@ -87,5 +89,5 @@ export declare function startWebServer(options: WebServerOptions): Promise<WebSe
87
89
  * @param port - Port to bind to (default: 3939)
88
90
  * @returns Result with URL, server status, and browser open status
89
91
  */
90
- export declare function openPortfolioBrowser(portfolioDir: string, port?: number, mcpAqlHandler?: MCPAQLHandler): Promise<BrowserOpenResult>;
92
+ export declare function openPortfolioBrowser(portfolioDir: string, port?: number, mcpAqlHandler?: MCPAQLHandler, tab?: string): Promise<BrowserOpenResult>;
91
93
  //# sourceMappingURL=server.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../src/web/server.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAcH,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,mCAAmC,CAAC;AACvE,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,uCAAuC,CAAC;AAW/E;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,sCAAsC;IACtC,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,qEAAqE;IACrE,YAAY,EAAE,MAAM,CAAC;IACrB,qEAAqE;IACrE,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB;;;;;OAKG;IACH,aAAa,CAAC,EAAE,aAAa,CAAC;IAC9B,kFAAkF;IAClF,UAAU,CAAC,EAAE,aAAa,CAAC;IAC3B,6FAA6F;IAC7F,WAAW,CAAC,EAAE,iBAAiB,CAAC;IAChC,gFAAgF;IAChF,iBAAiB,CAAC,EAAE,OAAO,SAAS,EAAE,MAAM,EAAE,CAAC;CAChD;AAED;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B,kFAAkF;IAClF,GAAG,CAAC,EAAE,OAAO,SAAS,EAAE,OAAO,CAAC;IAChC,2EAA2E;IAC3E,YAAY,CAAC,EAAE,CAAC,KAAK,EAAE,OAAO,qBAAqB,EAAE,eAAe,KAAK,IAAI,CAAC;IAC9E,iFAAiF;IACjF,iBAAiB,CAAC,EAAE,CAAC,QAAQ,EAAE,OAAO,qBAAqB,EAAE,cAAc,KAAK,IAAI,CAAC;CACtF;AAED;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,uCAAuC;IACvC,GAAG,EAAE,MAAM,CAAC;IACZ,4EAA4E;IAC5E,cAAc,EAAE,OAAO,CAAC;IACxB,kDAAkD;IAClD,aAAa,EAAE,OAAO,CAAC;IACvB,yDAAyD;IACzD,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAsCD;;;;;;;;;;;GAWG;AACH,wBAAsB,cAAc,CAAC,OAAO,EAAE,gBAAgB,GAAG,OAAO,CAAC,eAAe,CAAC,CAkJxF;AAED;;;;;;;;;;;;GAYG;AACH,wBAAsB,oBAAoB,CAAC,YAAY,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM,EAAE,aAAa,CAAC,EAAE,aAAa,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAwBzI"}
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../src/web/server.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAcH,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,mCAAmC,CAAC;AACvE,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,uCAAuC,CAAC;AAW/E,qEAAqE;AACrE,wBAAgB,kBAAkB,IAAI,OAAO,CAE5C;AAED;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,sCAAsC;IACtC,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,qEAAqE;IACrE,YAAY,EAAE,MAAM,CAAC;IACrB,qEAAqE;IACrE,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB;;;;;OAKG;IACH,aAAa,CAAC,EAAE,aAAa,CAAC;IAC9B,kFAAkF;IAClF,UAAU,CAAC,EAAE,aAAa,CAAC;IAC3B,6FAA6F;IAC7F,WAAW,CAAC,EAAE,iBAAiB,CAAC;IAChC,gFAAgF;IAChF,iBAAiB,CAAC,EAAE,OAAO,SAAS,EAAE,MAAM,EAAE,CAAC;CAChD;AAED;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B,kFAAkF;IAClF,GAAG,CAAC,EAAE,OAAO,SAAS,EAAE,OAAO,CAAC;IAChC,2EAA2E;IAC3E,YAAY,CAAC,EAAE,CAAC,KAAK,EAAE,OAAO,qBAAqB,EAAE,eAAe,KAAK,IAAI,CAAC;IAC9E,iFAAiF;IACjF,iBAAiB,CAAC,EAAE,CAAC,QAAQ,EAAE,OAAO,qBAAqB,EAAE,cAAc,KAAK,IAAI,CAAC;CACtF;AAED;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,uCAAuC;IACvC,GAAG,EAAE,MAAM,CAAC;IACZ,4EAA4E;IAC5E,cAAc,EAAE,OAAO,CAAC;IACxB,kDAAkD;IAClD,aAAa,EAAE,OAAO,CAAC;IACvB,yDAAyD;IACzD,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAsCD;;;;;;;;;;;GAWG;AACH,wBAAsB,cAAc,CAAC,OAAO,EAAE,gBAAgB,GAAG,OAAO,CAAC,eAAe,CAAC,CAsKxF;AAED;;;;;;;;;;;;GAYG;AACH,wBAAsB,oBAAoB,CAAC,YAAY,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM,EAAE,aAAa,CAAC,EAAE,aAAa,EAAE,GAAG,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAuBvJ"}
@@ -29,6 +29,10 @@ const ALLOWED_PAGE_EXTENSIONS = new Set(['.html', '.htm']);
29
29
  /** Track whether the web server is already running in-process. */
30
30
  let serverRunning = false;
31
31
  let serverPort = DEFAULT_PORT;
32
+ /** Check whether the web server has been started in this process. */
33
+ export function isWebServerRunning() {
34
+ return serverRunning;
35
+ }
32
36
  /**
33
37
  * Open a URL in the system's default browser.
34
38
  *
@@ -192,17 +196,38 @@ export async function startWebServer(options) {
192
196
  }
193
197
  res.sendFile(join(publicDir, 'index.html'));
194
198
  });
195
- // Bind to localhost only
196
- app.listen(port, '127.0.0.1', () => {
197
- serverRunning = true;
198
- serverPort = port;
199
- const url = `http://${CONSOLE_HOST}:${port}`;
200
- const fallbackUrl = `http://127.0.0.1:${port}`;
201
- logger.info(`[WebUI] Management console running at ${url}`);
202
- console.log(`\n DollhouseMCP Management Console\n ${url}\n ${fallbackUrl} (fallback)\n`);
203
- if (options.openBrowser) {
204
- openInBrowser(url);
205
- }
199
+ // Bind to localhost only — handle port conflicts gracefully
200
+ // NOTE: console.log is intentional here (not logger). In --web standalone mode,
201
+ // the user sees terminal output directly. logger.info writes to the structured
202
+ // log system (MemoryLogSink → SSE → web console). Both are needed: console.log
203
+ // for the human at the terminal, logger for the log viewer tab.
204
+ await new Promise((resolve) => {
205
+ const httpServer = app.listen(port, '127.0.0.1', () => {
206
+ serverRunning = true;
207
+ serverPort = port;
208
+ const url = `http://${CONSOLE_HOST}:${port}`;
209
+ const fallbackUrl = `http://127.0.0.1:${port}`;
210
+ logger.info(`[WebUI] Management console running at ${url}`);
211
+ console.log(`\n DollhouseMCP Management Console\n ${url}\n ${fallbackUrl} (fallback)\n`);
212
+ if (options.openBrowser) {
213
+ openInBrowser(url);
214
+ }
215
+ resolve();
216
+ });
217
+ httpServer.on('error', (err) => {
218
+ if (err.code === 'EADDRINUSE') {
219
+ const url = `http://${CONSOLE_HOST}:${port}`;
220
+ logger.info(`[WebUI] Port ${port} already in use — opening existing console`);
221
+ console.log(`\n DollhouseMCP Management Console (existing instance)\n ${url}\n`);
222
+ if (options.openBrowser) {
223
+ openInBrowser(url);
224
+ }
225
+ }
226
+ else {
227
+ logger.error(`[WebUI] Failed to bind port ${port}: ${err.message}`);
228
+ }
229
+ resolve(); // Web console is optional — don't block startup
230
+ });
206
231
  });
207
232
  return result;
208
233
  }
@@ -219,9 +244,10 @@ export async function startWebServer(options) {
219
244
  * @param port - Port to bind to (default: 3939)
220
245
  * @returns Result with URL, server status, and browser open status
221
246
  */
222
- export async function openPortfolioBrowser(portfolioDir, port, mcpAqlHandler) {
247
+ export async function openPortfolioBrowser(portfolioDir, port, mcpAqlHandler, tab) {
223
248
  const targetPort = port || DEFAULT_PORT;
224
- const url = `http://${CONSOLE_HOST}:${targetPort}`;
249
+ const baseUrl = `http://${CONSOLE_HOST}:${targetPort}`;
250
+ const url = tab ? `${baseUrl}/#${tab}` : baseUrl;
225
251
  const alreadyRunning = serverRunning;
226
252
  if (!serverRunning) {
227
253
  await startWebServer({
@@ -230,8 +256,6 @@ export async function openPortfolioBrowser(portfolioDir, port, mcpAqlHandler) {
230
256
  openBrowser: false, // We'll open manually below to capture the result
231
257
  mcpAqlHandler,
232
258
  });
233
- // Wait briefly for the server to bind
234
- await new Promise(resolve => setTimeout(resolve, 500));
235
259
  }
236
260
  const browserResult = await openInBrowser(url);
237
261
  return {
@@ -241,4 +265,4 @@ export async function openPortfolioBrowser(portfolioDir, port, mcpAqlHandler) {
241
265
  ...(browserResult.error ? { warning: `Browser could not be opened automatically: ${browserResult.error}. Open ${url} manually.` } : {}),
242
266
  };
243
267
  }
244
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/web/server.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * DollhouseMCP Web UI Server\n *\n * Lightweight Express server for browsing portfolio elements in a browser.\n * Bound to 127.0.0.1 only (localhost). Read-only for V1.\n *\n * Can be started standalone (`--web` flag) or from within the MCP server\n * process via `openPortfolioBrowser()`.\n *\n * @see https://github.com/DollhouseMCP/mcp-server-v2-refactor/issues/704\n * @see https://github.com/DollhouseMCP/mcp-server-v2-refactor/issues/774\n */\n\nimport express from 'express';\nimport { join, dirname, extname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { execFile } from 'node:child_process';\nimport { platform } from 'node:os';\nimport { mkdir, readdir } from 'node:fs/promises';\nimport { createApiRoutes, createGatewayApiRoutes } from './routes.js';\nimport { createLogRoutes, type LogRoutesResult } from './routes/logRoutes.js';\nimport { createMetricsRoutes, type MetricsRoutesResult } from './routes/metricsRoutes.js';\nimport { createHealthRoutes } from './routes/healthRoutes.js';\nimport { createSetupRoutes } from './routes/setupRoutes.js';\nimport { logger } from '../utils/logger.js';\nimport type { MCPAQLHandler } from '../handlers/mcp-aql/MCPAQLHandler.js';\nimport type { MemoryLogSink } from '../logging/sinks/MemoryLogSink.js';\nimport type { MemoryMetricsSink } from '../metrics/sinks/MemoryMetricsSink.js';\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\nconst DEFAULT_PORT = 3939;\nconst CONSOLE_HOST = 'dollhouse.localhost';\nconst ALLOWED_PAGE_EXTENSIONS = new Set(['.html', '.htm']);\n\n/** Track whether the web server is already running in-process. */\nlet serverRunning = false;\nlet serverPort = DEFAULT_PORT;\n\n/**\n * Options for starting the web server.\n */\nexport interface WebServerOptions {\n  /** Port to bind to (default: 3939) */\n  port?: number;\n  /** Path to the portfolio directory (e.g., ~/.dollhouse/portfolio) */\n  portfolioDir: string;\n  /** Open the browser automatically after starting (default: false) */\n  openBrowser?: boolean;\n  /**\n   * MCPAQLHandler for routing through the MCP-AQL pipeline.\n   * When provided, API routes use the gateway (validated, cached, gatekeeper-checked).\n   * When absent, falls back to direct filesystem access (legacy behavior).\n   * Issue #796: Web MCP-AQL Gateway.\n   */\n  mcpAqlHandler?: MCPAQLHandler;\n  /** MemoryLogSink for log routes (optional — logs tab disabled if not provided) */\n  memorySink?: MemoryLogSink;\n  /** MemoryMetricsSink for metrics routes (optional — metrics tab disabled if not provided) */\n  metricsSink?: MemoryMetricsSink;\n  /** Additional routers to mount before the SPA fallback (e.g., ingest routes) */\n  additionalRouters?: import('express').Router[];\n}\n\n/**\n * Result of starting the web server, including hooks for DI wiring.\n */\nexport interface WebServerResult {\n  /** Express app instance — for mounting additional routes (e.g., ingest routes) */\n  app?: import('express').Express;\n  /** Log broadcast function — call with each entry to push to SSE clients */\n  logBroadcast?: (entry: import('../logging/types.js').UnifiedLogEntry) => void;\n  /** Metrics snapshot function — call with each snapshot to push to SSE clients */\n  metricsOnSnapshot?: (snapshot: import('../metrics/types.js').MetricSnapshot) => void;\n}\n\n/**\n * Result of attempting to open the browser.\n */\nexport interface BrowserOpenResult {\n  /** The URL the server is running on */\n  url: string;\n  /** Whether the server was already running (true) or just started (false) */\n  alreadyRunning: boolean;\n  /** Whether the browser was successfully opened */\n  browserOpened: boolean;\n  /** Warning message if the browser could not be opened */\n  warning?: string;\n}\n\n/**\n * Open a URL in the system's default browser.\n *\n * Platform-aware:\n * - macOS: `open`\n * - Linux: `xdg-open`\n * - Windows: `start`\n *\n * @param url - The URL to open\n * @returns Promise that resolves to true if the browser opened, false with error message if not\n */\nfunction openInBrowser(url: string): Promise<{ success: boolean; error?: string }> {\n  return new Promise((resolve) => {\n    const plat = platform();\n    const cmd = plat === 'darwin' ? 'open'\n      : plat === 'win32' ? 'start'\n      : 'xdg-open';\n\n    // Security: use execFile with URL as argument array, not string interpolation\n    const urlStr = String(url);\n    // Accept localhost, 127.0.0.1, and *.localhost subdomains (RFC 6761)\n    if (!/^https?:\\/\\/(localhost|127\\.0\\.0\\.1|[\\w-]+\\.localhost)[:/]/.test(urlStr)) {\n      resolve({ success: false, error: 'URL must be a localhost HTTP URL' });\n      return;\n    }\n    execFile(cmd, [urlStr], (err) => {\n      if (err) {\n        logger.warn(`[WebUI] Could not auto-open browser: ${err.message}`);\n        resolve({ success: false, error: err.message });\n      } else {\n        resolve({ success: true });\n      }\n    });\n  });\n}\n\n/**\n * Start the portfolio web server.\n *\n * Binds to 127.0.0.1 only (localhost). Serves the portfolio browser\n * frontend and API routes for reading elements.\n *\n * Idempotent: if the server is already running, optionally opens the\n * browser without starting a second instance.\n *\n * @param options - Server configuration\n * @returns Hooks for DI wiring (log broadcast, metrics onSnapshot)\n */\nexport async function startWebServer(options: WebServerOptions): Promise<WebServerResult> {\n  const port = options.port || DEFAULT_PORT;\n  const result: WebServerResult = {};\n\n  if (serverRunning) {\n    if (options.openBrowser) {\n      openInBrowser(`http://${CONSOLE_HOST}:${serverPort}`);\n    }\n    return result;\n  }\n\n  const app = express();\n  result.app = app;\n  app.disable('x-powered-by');\n\n  // Security headers\n  app.use((_req, res, next) => {\n    res.setHeader('X-Content-Type-Options', 'nosniff');\n    res.setHeader('X-Frame-Options', 'DENY');\n    res.setHeader('X-XSS-Protection', '1; mode=block');\n    res.setHeader('Referrer-Policy', 'no-referrer');\n    res.setHeader('Access-Control-Allow-Origin', `http://${CONSOLE_HOST}:${port}`);\n    res.setHeader('Content-Security-Policy', [\n      \"default-src 'self'\",\n      \"script-src 'self' cdn.jsdelivr.net cdnjs.cloudflare.com\",\n      \"style-src 'self' 'unsafe-inline' cdnjs.cloudflare.com cdn.jsdelivr.net\",\n      \"connect-src 'self' raw.githubusercontent.com\",\n      \"font-src 'self'\",\n    ].join('; '));\n    next();\n  });\n\n  // Setup routes: auto-install DollhouseMCP to MCP clients (mount BEFORE API routes)\n  app.use(express.json({ limit: '1kb', type: 'application/json' }));\n  const { installHandler, openConfigHandler, versionHandler, mcpbRedirectHandler, detectHandler } = createSetupRoutes();\n  app.post('/api/setup/install', installHandler);\n  app.post('/api/setup/open-config', openConfigHandler);\n  app.get('/api/setup/version', versionHandler);\n  app.get('/api/setup/mcpb', mcpbRedirectHandler);\n  app.get('/api/setup/detect', detectHandler);\n  logger.info('[WebUI] Setup routes mounted at /api/setup');\n\n  // API routes — use MCP-AQL gateway when handler is available (Issue #796)\n  if (options.mcpAqlHandler) {\n    app.use('/api', createGatewayApiRoutes(options.mcpAqlHandler, options.portfolioDir));\n\n    // Permission evaluation routes (POST /evaluate_permission, GET /permissions/status)\n    const { registerPermissionRoutes } = await import('./routes/permissionRoutes.js');\n    const permRouter = (await import('express')).Router();\n    registerPermissionRoutes(permRouter, options.mcpAqlHandler);\n    app.use('/api', permRouter);\n\n    logger.info('[WebUI] API routes using MCP-AQL Gateway + permission routes');\n  } else {\n    app.use('/api', createApiRoutes(options.portfolioDir));\n    logger.warn('[WebUI] API routes using direct filesystem access (no MCP-AQL handler available)');\n  }\n\n  // Console routes: logs, metrics, health\n  let logRoutes: LogRoutesResult | undefined;\n  let metricsRoutes: MetricsRoutesResult | undefined;\n\n  if (options.memorySink) {\n    logRoutes = createLogRoutes(options.memorySink);\n    app.use('/api', logRoutes.router);\n    result.logBroadcast = logRoutes.broadcast;\n    logger.info('[WebUI] Log viewer routes mounted at /api/logs');\n  }\n\n  if (options.metricsSink) {\n    metricsRoutes = createMetricsRoutes(options.metricsSink);\n    app.use('/api', metricsRoutes.router);\n    result.metricsOnSnapshot = metricsRoutes.onSnapshot;\n    logger.info('[WebUI] Metrics routes mounted at /api/metrics');\n  }\n\n  if (options.memorySink) {\n    const healthRouter = createHealthRoutes({\n      memorySink: options.memorySink,\n      metricsSink: options.metricsSink,\n      logClientCount: logRoutes ? logRoutes.clientCount : () => 0,\n      metricsClientCount: metricsRoutes ? metricsRoutes.clientCount : () => 0,\n    });\n    app.use('/api', healthRouter);\n  }\n\n  // Serve ~/.dollhouse/pages/ at /pages/ — dashboards, generated content, stack views\n  const pagesDir = join(dirname(options.portfolioDir), 'pages');\n  mkdir(pagesDir, { recursive: true }).catch(err => {\n    logger.warn(`[WebUI] Could not create pages directory: ${(err as Error).message}`);\n  });\n  app.use('/pages', express.static(pagesDir));\n\n  /**\n   * GET /api/pages\n   * Lists available HTML pages in ~/.dollhouse/pages/.\n   * Returns page names and their URLs for the management console.\n   */\n  app.get('/api/pages', async (_req, res) => {\n    try {\n      const files = await readdir(pagesDir);\n      const pages = files\n        .filter(f => !f.startsWith('.') && ALLOWED_PAGE_EXTENSIONS.has(extname(f)))\n        .map(f => ({ name: f, url: `/pages/${f}` }));\n      res.json({ pages, directory: pagesDir });\n    } catch {\n      res.json({ pages: [], directory: pagesDir });\n    }\n  });\n\n  // Additional routers (e.g., unified console ingest routes) — must mount before SPA fallback\n  options.additionalRouters?.forEach(router => app.use(router));\n\n  // Static frontend files\n  const publicDir = join(__dirname, 'public');\n  app.use(express.static(publicDir));\n\n  // SPA fallback\n  app.get('/{*path}', (req, res) => {\n    const normalizedPath = req.path.normalize('NFC');\n    if (normalizedPath.startsWith('/api/')) {\n      res.status(404).json({ error: `API route not found: ${normalizedPath}` });\n      return;\n    }\n    if (normalizedPath.startsWith('/pages/')) {\n      res.status(404).json({ error: `Page not found: ${normalizedPath}` });\n      return;\n    }\n    res.sendFile(join(publicDir, 'index.html'));\n  });\n\n  // Bind to localhost only\n  app.listen(port, '127.0.0.1', () => {\n    serverRunning = true;\n    serverPort = port;\n    const url = `http://${CONSOLE_HOST}:${port}`;\n    const fallbackUrl = `http://127.0.0.1:${port}`;\n    logger.info(`[WebUI] Management console running at ${url}`);\n    console.log(`\\n  DollhouseMCP Management Console\\n  ${url}\\n  ${fallbackUrl} (fallback)\\n`);\n\n    if (options.openBrowser) {\n      openInBrowser(url);\n    }\n  });\n\n  return result;\n}\n\n/**\n * Open the portfolio browser from within the MCP server process.\n *\n * Starts the web server if not already running, then opens the system\n * browser to the portfolio UI. Returns a result object indicating\n * whether the server started and the browser opened successfully.\n *\n * Called by the `open_portfolio_browser` MCP-AQL operation (Issue #774).\n *\n * @param portfolioDir - Path to the portfolio directory (e.g., ~/.dollhouse/portfolio)\n * @param port - Port to bind to (default: 3939)\n * @returns Result with URL, server status, and browser open status\n */\nexport async function openPortfolioBrowser(portfolioDir: string, port?: number, mcpAqlHandler?: MCPAQLHandler): Promise<BrowserOpenResult> {\n  const targetPort = port || DEFAULT_PORT;\n  const url = `http://${CONSOLE_HOST}:${targetPort}`;\n  const alreadyRunning = serverRunning;\n\n  if (!serverRunning) {\n    await startWebServer({\n      portfolioDir,\n      port: targetPort,\n      openBrowser: false, // We'll open manually below to capture the result\n      mcpAqlHandler,\n    });\n    // Wait briefly for the server to bind\n    await new Promise(resolve => setTimeout(resolve, 500));\n  }\n\n  const browserResult = await openInBrowser(url);\n\n  return {\n    url,\n    alreadyRunning,\n    browserOpened: browserResult.success,\n    ...(browserResult.error ? { warning: `Browser could not be opened automatically: ${browserResult.error}. Open ${url} manually.` } : {}),\n  };\n}\n"]}
268
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/web/server.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * DollhouseMCP Web UI Server\n *\n * Lightweight Express server for browsing portfolio elements in a browser.\n * Bound to 127.0.0.1 only (localhost). Read-only for V1.\n *\n * Can be started standalone (`--web` flag) or from within the MCP server\n * process via `openPortfolioBrowser()`.\n *\n * @see https://github.com/DollhouseMCP/mcp-server-v2-refactor/issues/704\n * @see https://github.com/DollhouseMCP/mcp-server-v2-refactor/issues/774\n */\n\nimport express from 'express';\nimport { join, dirname, extname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { execFile } from 'node:child_process';\nimport { platform } from 'node:os';\nimport { mkdir, readdir } from 'node:fs/promises';\nimport { createApiRoutes, createGatewayApiRoutes } from './routes.js';\nimport { createLogRoutes, type LogRoutesResult } from './routes/logRoutes.js';\nimport { createMetricsRoutes, type MetricsRoutesResult } from './routes/metricsRoutes.js';\nimport { createHealthRoutes } from './routes/healthRoutes.js';\nimport { createSetupRoutes } from './routes/setupRoutes.js';\nimport { logger } from '../utils/logger.js';\nimport type { MCPAQLHandler } from '../handlers/mcp-aql/MCPAQLHandler.js';\nimport type { MemoryLogSink } from '../logging/sinks/MemoryLogSink.js';\nimport type { MemoryMetricsSink } from '../metrics/sinks/MemoryMetricsSink.js';\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\nconst DEFAULT_PORT = 3939;\nconst CONSOLE_HOST = 'dollhouse.localhost';\nconst ALLOWED_PAGE_EXTENSIONS = new Set(['.html', '.htm']);\n\n/** Track whether the web server is already running in-process. */\nlet serverRunning = false;\nlet serverPort = DEFAULT_PORT;\n\n/** Check whether the web server has been started in this process. */\nexport function isWebServerRunning(): boolean {\n  return serverRunning;\n}\n\n/**\n * Options for starting the web server.\n */\nexport interface WebServerOptions {\n  /** Port to bind to (default: 3939) */\n  port?: number;\n  /** Path to the portfolio directory (e.g., ~/.dollhouse/portfolio) */\n  portfolioDir: string;\n  /** Open the browser automatically after starting (default: false) */\n  openBrowser?: boolean;\n  /**\n   * MCPAQLHandler for routing through the MCP-AQL pipeline.\n   * When provided, API routes use the gateway (validated, cached, gatekeeper-checked).\n   * When absent, falls back to direct filesystem access (legacy behavior).\n   * Issue #796: Web MCP-AQL Gateway.\n   */\n  mcpAqlHandler?: MCPAQLHandler;\n  /** MemoryLogSink for log routes (optional — logs tab disabled if not provided) */\n  memorySink?: MemoryLogSink;\n  /** MemoryMetricsSink for metrics routes (optional — metrics tab disabled if not provided) */\n  metricsSink?: MemoryMetricsSink;\n  /** Additional routers to mount before the SPA fallback (e.g., ingest routes) */\n  additionalRouters?: import('express').Router[];\n}\n\n/**\n * Result of starting the web server, including hooks for DI wiring.\n */\nexport interface WebServerResult {\n  /** Express app instance — for mounting additional routes (e.g., ingest routes) */\n  app?: import('express').Express;\n  /** Log broadcast function — call with each entry to push to SSE clients */\n  logBroadcast?: (entry: import('../logging/types.js').UnifiedLogEntry) => void;\n  /** Metrics snapshot function — call with each snapshot to push to SSE clients */\n  metricsOnSnapshot?: (snapshot: import('../metrics/types.js').MetricSnapshot) => void;\n}\n\n/**\n * Result of attempting to open the browser.\n */\nexport interface BrowserOpenResult {\n  /** The URL the server is running on */\n  url: string;\n  /** Whether the server was already running (true) or just started (false) */\n  alreadyRunning: boolean;\n  /** Whether the browser was successfully opened */\n  browserOpened: boolean;\n  /** Warning message if the browser could not be opened */\n  warning?: string;\n}\n\n/**\n * Open a URL in the system's default browser.\n *\n * Platform-aware:\n * - macOS: `open`\n * - Linux: `xdg-open`\n * - Windows: `start`\n *\n * @param url - The URL to open\n * @returns Promise that resolves to true if the browser opened, false with error message if not\n */\nfunction openInBrowser(url: string): Promise<{ success: boolean; error?: string }> {\n  return new Promise((resolve) => {\n    const plat = platform();\n    const cmd = plat === 'darwin' ? 'open'\n      : plat === 'win32' ? 'start'\n      : 'xdg-open';\n\n    // Security: use execFile with URL as argument array, not string interpolation\n    const urlStr = String(url);\n    // Accept localhost, 127.0.0.1, and *.localhost subdomains (RFC 6761)\n    if (!/^https?:\\/\\/(localhost|127\\.0\\.0\\.1|[\\w-]+\\.localhost)[:/]/.test(urlStr)) {\n      resolve({ success: false, error: 'URL must be a localhost HTTP URL' });\n      return;\n    }\n    execFile(cmd, [urlStr], (err) => {\n      if (err) {\n        logger.warn(`[WebUI] Could not auto-open browser: ${err.message}`);\n        resolve({ success: false, error: err.message });\n      } else {\n        resolve({ success: true });\n      }\n    });\n  });\n}\n\n/**\n * Start the portfolio web server.\n *\n * Binds to 127.0.0.1 only (localhost). Serves the portfolio browser\n * frontend and API routes for reading elements.\n *\n * Idempotent: if the server is already running, optionally opens the\n * browser without starting a second instance.\n *\n * @param options - Server configuration\n * @returns Hooks for DI wiring (log broadcast, metrics onSnapshot)\n */\nexport async function startWebServer(options: WebServerOptions): Promise<WebServerResult> {\n  const port = options.port || DEFAULT_PORT;\n  const result: WebServerResult = {};\n\n  if (serverRunning) {\n    if (options.openBrowser) {\n      openInBrowser(`http://${CONSOLE_HOST}:${serverPort}`);\n    }\n    return result;\n  }\n\n  const app = express();\n  result.app = app;\n  app.disable('x-powered-by');\n\n  // Security headers\n  app.use((_req, res, next) => {\n    res.setHeader('X-Content-Type-Options', 'nosniff');\n    res.setHeader('X-Frame-Options', 'DENY');\n    res.setHeader('X-XSS-Protection', '1; mode=block');\n    res.setHeader('Referrer-Policy', 'no-referrer');\n    res.setHeader('Access-Control-Allow-Origin', `http://${CONSOLE_HOST}:${port}`);\n    res.setHeader('Content-Security-Policy', [\n      \"default-src 'self'\",\n      \"script-src 'self' cdn.jsdelivr.net cdnjs.cloudflare.com\",\n      \"style-src 'self' 'unsafe-inline' cdnjs.cloudflare.com cdn.jsdelivr.net\",\n      \"connect-src 'self' raw.githubusercontent.com\",\n      \"font-src 'self'\",\n    ].join('; '));\n    next();\n  });\n\n  // Setup routes: auto-install DollhouseMCP to MCP clients (mount BEFORE API routes)\n  app.use(express.json({ limit: '1kb', type: 'application/json' }));\n  const { installHandler, openConfigHandler, versionHandler, mcpbRedirectHandler, detectHandler } = createSetupRoutes();\n  app.post('/api/setup/install', installHandler);\n  app.post('/api/setup/open-config', openConfigHandler);\n  app.get('/api/setup/version', versionHandler);\n  app.get('/api/setup/mcpb', mcpbRedirectHandler);\n  app.get('/api/setup/detect', detectHandler);\n  logger.info('[WebUI] Setup routes mounted at /api/setup');\n\n  // API routes — use MCP-AQL gateway when handler is available (Issue #796)\n  if (options.mcpAqlHandler) {\n    app.use('/api', createGatewayApiRoutes(options.mcpAqlHandler, options.portfolioDir));\n\n    // Permission evaluation routes (POST /evaluate_permission, GET /permissions/status)\n    const { registerPermissionRoutes } = await import('./routes/permissionRoutes.js');\n    const permRouter = (await import('express')).Router();\n    registerPermissionRoutes(permRouter, options.mcpAqlHandler);\n    app.use('/api', permRouter);\n\n    logger.info('[WebUI] API routes using MCP-AQL Gateway + permission routes');\n  } else {\n    app.use('/api', createApiRoutes(options.portfolioDir));\n    logger.warn('[WebUI] API routes using direct filesystem access (no MCP-AQL handler available)');\n  }\n\n  // Console routes: logs, metrics, health\n  let logRoutes: LogRoutesResult | undefined;\n  let metricsRoutes: MetricsRoutesResult | undefined;\n\n  if (options.memorySink) {\n    logRoutes = createLogRoutes(options.memorySink);\n    app.use('/api', logRoutes.router);\n    result.logBroadcast = logRoutes.broadcast;\n    logger.info('[WebUI] Log viewer routes mounted at /api/logs');\n  }\n\n  if (options.metricsSink) {\n    metricsRoutes = createMetricsRoutes(options.metricsSink);\n    app.use('/api', metricsRoutes.router);\n    result.metricsOnSnapshot = metricsRoutes.onSnapshot;\n    logger.info('[WebUI] Metrics routes mounted at /api/metrics');\n  }\n\n  if (options.memorySink) {\n    const healthRouter = createHealthRoutes({\n      memorySink: options.memorySink,\n      metricsSink: options.metricsSink,\n      logClientCount: logRoutes ? logRoutes.clientCount : () => 0,\n      metricsClientCount: metricsRoutes ? metricsRoutes.clientCount : () => 0,\n    });\n    app.use('/api', healthRouter);\n  }\n\n  // Serve ~/.dollhouse/pages/ at /pages/ — dashboards, generated content, stack views\n  const pagesDir = join(dirname(options.portfolioDir), 'pages');\n  mkdir(pagesDir, { recursive: true }).catch(err => {\n    logger.warn(`[WebUI] Could not create pages directory: ${(err as Error).message}`);\n  });\n  app.use('/pages', express.static(pagesDir));\n\n  /**\n   * GET /api/pages\n   * Lists available HTML pages in ~/.dollhouse/pages/.\n   * Returns page names and their URLs for the management console.\n   */\n  app.get('/api/pages', async (_req, res) => {\n    try {\n      const files = await readdir(pagesDir);\n      const pages = files\n        .filter(f => !f.startsWith('.') && ALLOWED_PAGE_EXTENSIONS.has(extname(f)))\n        .map(f => ({ name: f, url: `/pages/${f}` }));\n      res.json({ pages, directory: pagesDir });\n    } catch {\n      res.json({ pages: [], directory: pagesDir });\n    }\n  });\n\n  // Additional routers (e.g., unified console ingest routes) — must mount before SPA fallback\n  options.additionalRouters?.forEach(router => app.use(router));\n\n  // Static frontend files\n  const publicDir = join(__dirname, 'public');\n  app.use(express.static(publicDir));\n\n  // SPA fallback\n  app.get('/{*path}', (req, res) => {\n    const normalizedPath = req.path.normalize('NFC');\n    if (normalizedPath.startsWith('/api/')) {\n      res.status(404).json({ error: `API route not found: ${normalizedPath}` });\n      return;\n    }\n    if (normalizedPath.startsWith('/pages/')) {\n      res.status(404).json({ error: `Page not found: ${normalizedPath}` });\n      return;\n    }\n    res.sendFile(join(publicDir, 'index.html'));\n  });\n\n  // Bind to localhost only — handle port conflicts gracefully\n  // NOTE: console.log is intentional here (not logger). In --web standalone mode,\n  // the user sees terminal output directly. logger.info writes to the structured\n  // log system (MemoryLogSink → SSE → web console). Both are needed: console.log\n  // for the human at the terminal, logger for the log viewer tab.\n  await new Promise<void>((resolve) => {\n    const httpServer = app.listen(port, '127.0.0.1', () => {\n      serverRunning = true;\n      serverPort = port;\n      const url = `http://${CONSOLE_HOST}:${port}`;\n      const fallbackUrl = `http://127.0.0.1:${port}`;\n      logger.info(`[WebUI] Management console running at ${url}`);\n      console.log(`\\n  DollhouseMCP Management Console\\n  ${url}\\n  ${fallbackUrl} (fallback)\\n`);\n\n      if (options.openBrowser) {\n        openInBrowser(url);\n      }\n      resolve();\n    });\n    httpServer.on('error', (err: NodeJS.ErrnoException) => {\n      if (err.code === 'EADDRINUSE') {\n        const url = `http://${CONSOLE_HOST}:${port}`;\n        logger.info(`[WebUI] Port ${port} already in use — opening existing console`);\n        console.log(`\\n  DollhouseMCP Management Console (existing instance)\\n  ${url}\\n`);\n        if (options.openBrowser) {\n          openInBrowser(url);\n        }\n      } else {\n        logger.error(`[WebUI] Failed to bind port ${port}: ${err.message}`);\n      }\n      resolve(); // Web console is optional — don't block startup\n    });\n  });\n\n  return result;\n}\n\n/**\n * Open the portfolio browser from within the MCP server process.\n *\n * Starts the web server if not already running, then opens the system\n * browser to the portfolio UI. Returns a result object indicating\n * whether the server started and the browser opened successfully.\n *\n * Called by the `open_portfolio_browser` MCP-AQL operation (Issue #774).\n *\n * @param portfolioDir - Path to the portfolio directory (e.g., ~/.dollhouse/portfolio)\n * @param port - Port to bind to (default: 3939)\n * @returns Result with URL, server status, and browser open status\n */\nexport async function openPortfolioBrowser(portfolioDir: string, port?: number, mcpAqlHandler?: MCPAQLHandler, tab?: string): Promise<BrowserOpenResult> {\n  const targetPort = port || DEFAULT_PORT;\n  const baseUrl = `http://${CONSOLE_HOST}:${targetPort}`;\n  const url = tab ? `${baseUrl}/#${tab}` : baseUrl;\n  const alreadyRunning = serverRunning;\n\n  if (!serverRunning) {\n    await startWebServer({\n      portfolioDir,\n      port: targetPort,\n      openBrowser: false, // We'll open manually below to capture the result\n      mcpAqlHandler,\n    });\n  }\n\n  const browserResult = await openInBrowser(url);\n\n  return {\n    url,\n    alreadyRunning,\n    browserOpened: browserResult.success,\n    ...(browserResult.error ? { warning: `Browser could not be opened automatically: ${browserResult.error}. Open ${url} manually.` } : {}),\n  };\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dollhousemcp/mcp-server",
3
- "version": "2.0.5",
3
+ "version": "2.0.7",
4
4
  "description": "DollhouseMCP - A Model Context Protocol (MCP) server that enables dynamic AI persona management from markdown files, allowing Claude and other compatible AI assistants to activate and switch between different behavioral personas.",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",
package/server.json CHANGED
@@ -3,7 +3,7 @@
3
3
  "name": "io.github.DollhouseMCP/mcp-server",
4
4
  "title": "DollhouseMCP",
5
5
  "description": "OSS to create Personas, Skills, Templates, Agents, and Memories to customize your AI experience.",
6
- "version": "2.0.5",
6
+ "version": "2.0.7",
7
7
  "homepage": "https://dollhousemcp.com",
8
8
  "repository": {
9
9
  "type": "git",
@@ -29,7 +29,7 @@
29
29
  {
30
30
  "registryType": "npm",
31
31
  "identifier": "@dollhousemcp/mcp-server",
32
- "version": "2.0.5",
32
+ "version": "2.0.7",
33
33
  "transport": {
34
34
  "type": "stdio"
35
35
  }