@dollhousemcp/mcp-server 2.0.26 → 2.0.27-rc.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +4 -0
- package/dist/elements/ensembles/EnsembleManager.d.ts +18 -0
- package/dist/elements/ensembles/EnsembleManager.d.ts.map +1 -1
- package/dist/elements/ensembles/EnsembleManager.js +37 -5
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.d.ts.map +1 -1
- package/dist/generated/version.js +3 -3
- package/dist/logging/LogHooks.d.ts.map +1 -1
- package/dist/logging/LogHooks.js +33 -1
- package/dist/web/contentPipeline.d.ts +1 -0
- package/dist/web/contentPipeline.d.ts.map +1 -1
- package/dist/web/contentPipeline.js +48 -4
- package/dist/web/public/app.js +204 -83
- package/dist/web/public/index.html +18 -6
- package/dist/web/public/logs.css +157 -1
- package/dist/web/public/logs.js +56 -8
- package/dist/web/public/sessions.css +2 -1
- package/dist/web/public/styles.css +272 -20
- package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
- package/dist/web/routes/permissionRoutes.js +18 -4
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +13 -3
- package/package.json +1 -1
- package/server.json +2 -2
package/dist/logging/LogHooks.js
CHANGED
|
@@ -12,8 +12,10 @@
|
|
|
12
12
|
* cleanups.forEach(fn => fn());
|
|
13
13
|
*/
|
|
14
14
|
import { SecurityMonitor } from '../security/securityMonitor.js';
|
|
15
|
+
import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
|
|
15
16
|
import { DefaultElementProvider } from '../portfolio/DefaultElementProvider.js';
|
|
16
17
|
import { LRUCache } from '../cache/LRUCache.js';
|
|
18
|
+
import { EventDeduplicator } from '../utils/EventDeduplicator.js';
|
|
17
19
|
// ---------------------------------------------------------------------------
|
|
18
20
|
// Severity → LogLevel helper (shared by SecurityMonitor, SecurityTelemetry,
|
|
19
21
|
// SecurityAuditor)
|
|
@@ -28,6 +30,16 @@ const SEVERITY_TO_LEVEL = {
|
|
|
28
30
|
medium: 'warn',
|
|
29
31
|
low: 'info',
|
|
30
32
|
};
|
|
33
|
+
const CONTENT_INJECTION_VISIBILITY_WINDOW_MS = 10 * 60_000;
|
|
34
|
+
const CONTENT_INJECTION_VISIBILITY_MAX_KEYS = 500;
|
|
35
|
+
function extractContentInjectionKey(value) {
|
|
36
|
+
if (typeof value !== 'string') {
|
|
37
|
+
return null;
|
|
38
|
+
}
|
|
39
|
+
const normalizedValue = UnicodeValidator.normalize(value).normalizedContent;
|
|
40
|
+
const normalized = normalizedValue.replace(/^Detected pattern:\s*/i, '').trim();
|
|
41
|
+
return normalized === '' ? null : `CONTENT_INJECTION\0${normalized}`;
|
|
42
|
+
}
|
|
31
43
|
// ---------------------------------------------------------------------------
|
|
32
44
|
// Exported factory for TriggerMetricsTracker (created outside DI container)
|
|
33
45
|
// ---------------------------------------------------------------------------
|
|
@@ -76,6 +88,7 @@ export function getSecurityAuditorLogListener(logManager, contextTracker) {
|
|
|
76
88
|
*/
|
|
77
89
|
export function wireLogHooks(logManager, container) {
|
|
78
90
|
const cleanups = [];
|
|
91
|
+
const contentInjectionVisibilityDedup = new EventDeduplicator(CONTENT_INJECTION_VISIBILITY_WINDOW_MS, CONTENT_INJECTION_VISIBILITY_MAX_KEYS);
|
|
79
92
|
// Resolve ContextTracker for correlationId injection
|
|
80
93
|
let contextTracker = null;
|
|
81
94
|
try {
|
|
@@ -86,6 +99,13 @@ export function wireLogHooks(logManager, container) {
|
|
|
86
99
|
try {
|
|
87
100
|
const mcpLogger = container.resolve('MCPLogger');
|
|
88
101
|
const unsub = mcpLogger.addLogListener((logEntry) => {
|
|
102
|
+
const contentInjectionKey = logEntry.message === '[CRITICAL SECURITY ALERT]' &&
|
|
103
|
+
logEntry.data?.type === 'CONTENT_INJECTION_ATTEMPT'
|
|
104
|
+
? extractContentInjectionKey(logEntry.data?.details)
|
|
105
|
+
: null;
|
|
106
|
+
if (contentInjectionKey !== null && contentInjectionVisibilityDedup.shouldSuppress(contentInjectionKey)) {
|
|
107
|
+
return;
|
|
108
|
+
}
|
|
89
109
|
const entry = {
|
|
90
110
|
id: logManager.generateId(),
|
|
91
111
|
timestamp: logEntry.timestamp.toISOString(),
|
|
@@ -104,6 +124,12 @@ export function wireLogHooks(logManager, container) {
|
|
|
104
124
|
// --- SecurityMonitor (security, static) ---------------------------------
|
|
105
125
|
{
|
|
106
126
|
const unsub = SecurityMonitor.addLogListener((logEntry) => {
|
|
127
|
+
const contentInjectionKey = logEntry.type === 'CONTENT_INJECTION_ATTEMPT'
|
|
128
|
+
? extractContentInjectionKey(logEntry.details)
|
|
129
|
+
: null;
|
|
130
|
+
if (contentInjectionKey !== null && contentInjectionVisibilityDedup.shouldSuppress(contentInjectionKey)) {
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
107
133
|
const entry = {
|
|
108
134
|
id: logManager.generateId(),
|
|
109
135
|
timestamp: logEntry.timestamp,
|
|
@@ -127,6 +153,12 @@ export function wireLogHooks(logManager, container) {
|
|
|
127
153
|
try {
|
|
128
154
|
const secTelemetry = container.resolve('SecurityTelemetry');
|
|
129
155
|
const unsub = secTelemetry.addLogListener((telEntry) => {
|
|
156
|
+
const contentInjectionKey = telEntry.attackType === 'CONTENT_INJECTION'
|
|
157
|
+
? extractContentInjectionKey(telEntry.pattern)
|
|
158
|
+
: null;
|
|
159
|
+
if (contentInjectionKey !== null && contentInjectionVisibilityDedup.shouldSuppress(contentInjectionKey)) {
|
|
160
|
+
return;
|
|
161
|
+
}
|
|
130
162
|
const entry = {
|
|
131
163
|
id: logManager.generateId(),
|
|
132
164
|
timestamp: telEntry.timestamp,
|
|
@@ -300,4 +332,4 @@ export function wireLogHooks(logManager, container) {
|
|
|
300
332
|
catch { /* StateChangeNotifier not registered */ }
|
|
301
333
|
return cleanups;
|
|
302
334
|
}
|
|
303
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LogHooks.js","sourceRoot":"","sources":["../../src/logging/LogHooks.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Phase 4: Integration Hooks for Unified Logging System.\n *\n * Maps native events from 10 monitoring/logging systems into UnifiedLogEntry\n * objects and routes them through LogManager.  Each source system exposes a\n * lightweight `addLogListener` callback; the translation logic lives here so\n * source files stay minimal and never import UnifiedLogEntry.\n *\n * Usage:\n *   const cleanups = wireLogHooks(logManager, container);\n *   // later …\n *   cleanups.forEach(fn => fn());\n */\n\nimport type { LogManager } from './LogManager.js';\nimport type { LogLevel, UnifiedLogEntry } from './types.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { DefaultElementProvider } from '../portfolio/DefaultElementProvider.js';\nimport { LRUCache } from '../cache/LRUCache.js';\n\n// ---------------------------------------------------------------------------\n// Severity → LogLevel helper (shared by SecurityMonitor, SecurityTelemetry,\n// SecurityAuditor)\n// ---------------------------------------------------------------------------\n\nconst SEVERITY_TO_LEVEL: Record<string, LogLevel> = {\n  CRITICAL: 'error',\n  HIGH: 'error',\n  MEDIUM: 'warn',\n  LOW: 'info',\n  critical: 'error',\n  high: 'error',\n  medium: 'warn',\n  low: 'info',\n};\n\n// ---------------------------------------------------------------------------\n// CorrelationId provider interface (subset of ContextTracker)\n// ---------------------------------------------------------------------------\n\ntype CorrelationIdProvider = { getCorrelationId(): string | undefined };\n\n// ---------------------------------------------------------------------------\n// Exported factory for TriggerMetricsTracker (created outside DI container)\n// ---------------------------------------------------------------------------\n\nexport function getTriggerMetricsLogListener(\n  logManager: LogManager,\n  contextTracker?: CorrelationIdProvider,\n): (level: 'debug' | 'info' | 'warn' | 'error', message: string, data?: Record<string, unknown>) => void {\n  return (level, message, data) => {\n    const entry: UnifiedLogEntry = {\n      id: logManager.generateId(),\n      timestamp: new Date().toISOString(),\n      category: 'telemetry',\n      level,\n      source: 'TriggerMetricsTracker',\n      message,\n      data,\n      correlationId: contextTracker?.getCorrelationId(),\n    };\n    logManager.log(entry);\n  };\n}\n\n// ---------------------------------------------------------------------------\n// Exported factory for SecurityAuditor (created outside DI container)\n// ---------------------------------------------------------------------------\n\nexport function getSecurityAuditorLogListener(\n  logManager: LogManager,\n  contextTracker?: CorrelationIdProvider,\n): (level: 'debug' | 'info' | 'warn' | 'error', message: string, data?: Record<string, unknown>) => void {\n  return (level, message, data) => {\n    const entry: UnifiedLogEntry = {\n      id: logManager.generateId(),\n      timestamp: new Date().toISOString(),\n      category: 'security',\n      level,\n      source: 'SecurityAuditor',\n      message,\n      data,\n      correlationId: contextTracker?.getCorrelationId(),\n    };\n    logManager.log(entry);\n  };\n}\n\n// ---------------------------------------------------------------------------\n// Main wiring function\n// ---------------------------------------------------------------------------\n\n/**\n * Wire all monitoring systems into the unified logging pipeline.\n *\n * @param logManager  The LogManager singleton from the DI container.\n * @param container   The DI container (used to resolve source services).\n * @returns Array of unsubscribe functions — call them during shutdown.\n */\nexport function wireLogHooks(\n  logManager: LogManager,\n  container: { resolve<T>(name: string): T },\n): (() => void)[] {\n  const cleanups: (() => void)[] = [];\n\n  // Resolve ContextTracker for correlationId injection\n  let contextTracker: CorrelationIdProvider | null = null;\n  try {\n    contextTracker = container.resolve<CorrelationIdProvider>('ContextTracker');\n  } catch { /* ContextTracker not registered */ }\n\n  // --- MCPLogger (application) -------------------------------------------\n  try {\n    const mcpLogger = container.resolve<{\n      addLogListener(fn: (entry: { timestamp: Date; level: string; message: string; data?: any }) => void): () => void;\n    }>('MCPLogger');\n    const unsub = mcpLogger.addLogListener((logEntry) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: logEntry.timestamp.toISOString(),\n        category: 'application',\n        level: logEntry.level as LogLevel,\n        source: 'MCPLogger',\n        message: logEntry.message,\n        data: logEntry.data != null ? logEntry.data : undefined,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* MCPLogger not registered */ }\n\n  // --- SecurityMonitor (security, static) ---------------------------------\n  {\n    const unsub = SecurityMonitor.addLogListener((logEntry) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: logEntry.timestamp,\n        category: 'security',\n        level: SEVERITY_TO_LEVEL[logEntry.severity] ?? 'info',\n        source: 'SecurityMonitor',\n        message: `[${logEntry.type}] ${logEntry.details}`,\n        data: {\n          ...logEntry.additionalData,\n          eventType: logEntry.type,\n          severity: logEntry.severity,\n          sourceComponent: logEntry.source,\n        },\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  }\n\n  // --- SecurityTelemetry (security) ---------------------------------------\n  try {\n    const secTelemetry = container.resolve<{\n      addLogListener(fn: (entry: {\n        timestamp: string; attackType: string; pattern: string;\n        severity: string; source: string; metadata?: Record<string, any>;\n      }) => void): () => void;\n    }>('SecurityTelemetry');\n    const unsub = secTelemetry.addLogListener((telEntry) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: telEntry.timestamp,\n        category: 'security',\n        level: SEVERITY_TO_LEVEL[telEntry.severity] ?? 'info',\n        source: 'SecurityTelemetry',\n        message: `Blocked ${telEntry.attackType}: ${telEntry.pattern}`,\n        data: telEntry.metadata,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* SecurityTelemetry not registered */ }\n\n  // --- PerformanceMonitor (performance) -----------------------------------\n  try {\n    const perfMonitor = container.resolve<{\n      addLogListener(fn: (level: string, message: string, data?: Record<string, unknown>) => void): () => void;\n    }>('PerformanceMonitor');\n    const unsub = perfMonitor.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'performance',\n        level: level as LogLevel,\n        source: 'PerformanceMonitor',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* PerformanceMonitor not registered */ }\n\n  // --- ElementEventDispatcher (application) — already has on() -----------\n  try {\n    const dispatcher = container.resolve<{\n      on(event: string, handler: (payload: any) => void): () => void;\n    }>('ElementEventDispatcher');\n\n    const eventLevelMap: Record<string, LogLevel> = {\n      'element:load:error': 'error',\n      'element:save:error': 'error',\n      'element:delete:error': 'error',\n      'element:lock-timeout': 'warn',\n      'element:activate': 'info',\n      'element:deactivate': 'info',\n      'element:load:success': 'info',\n      'element:save:success': 'info',\n      'element:delete:success': 'info',\n    };\n\n    // Only log events that have a mapped level (errors, warnings, success, activate/deactivate).\n    // Skip start/cache/external-change events — they fire per-element and create noise\n    // without adding value beyond the completion/error logs.\n    const loggedEvents = Object.keys(eventLevelMap);\n\n    for (const eventName of loggedEvents) {\n      const unsub = dispatcher.on(eventName, (payload: any) => {\n        const level = eventLevelMap[eventName] ?? 'debug';\n        const requestCorrelationId = contextTracker?.getCorrelationId();\n        // Use elementId if available, fall back to filename from filePath\n        const elementName = payload.elementId\n          || (payload.filePath ? payload.filePath.replace(/\\.[^.]+$/, '') : '');\n        const entry: UnifiedLogEntry = {\n          id: logManager.generateId(),\n          timestamp: new Date().toISOString(),\n          category: 'application',\n          level,\n          source: 'ElementEventDispatcher',\n          message: `${eventName} [${payload.elementType ?? 'unknown'}:${elementName}]`,\n          data: {\n            ...payload.extra,\n            ...(payload.correlationId ? { operationId: payload.correlationId } : {}),\n            ...(payload.filePath ? { filePath: payload.filePath } : {}),\n          },\n          correlationId: requestCorrelationId ?? payload.correlationId,\n        };\n        logManager.log(entry);\n      });\n      cleanups.push(unsub);\n    }\n  } catch { /* ElementEventDispatcher not registered */ }\n\n  // --- OperationalTelemetry (telemetry) -----------------------------------\n  try {\n    const opsTelemetry = container.resolve<{\n      addLogListener(fn: (level: string, message: string, data?: Record<string, unknown>) => void): () => void;\n    }>('OperationalTelemetry');\n    const unsub = opsTelemetry.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'telemetry',\n        level: level as LogLevel,\n        source: 'OperationalTelemetry',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* OperationalTelemetry not registered */ }\n\n  // --- FileLockManager (performance) --------------------------------------\n  try {\n    const lockManager = container.resolve<{\n      addLogListener(fn: (level: string, message: string, data?: Record<string, unknown>) => void): () => void;\n    }>('FileLockManager');\n    const unsub = lockManager.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'performance',\n        level: level as LogLevel,\n        source: 'FileLockManager',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* FileLockManager not registered */ }\n\n  // --- DefaultElementProvider (performance, static) -----------------------\n  {\n    const unsub = DefaultElementProvider.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'performance',\n        level: level as LogLevel,\n        source: 'DefaultElementProvider',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  }\n\n  // --- LRUCache (performance, static) ------------------------------------\n  {\n    const unsub = LRUCache.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'performance',\n        level: level as LogLevel,\n        source: 'LRUCache',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  }\n\n  // --- StateChangeNotifier (application) — extends EventEmitter ----------\n  try {\n    const notifier = container.resolve<{\n      on(event: string, handler: (...args: any[]) => void): any;\n      removeListener(event: string, handler: (...args: any[]) => void): any;\n    }>('StateChangeNotifier');\n    const handler = (event: { type: string; previousValue: string | null; newValue: string | null }) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'application',\n        level: 'info',\n        source: 'StateChangeNotifier',\n        message: `State change: ${event.type}`,\n        data: { previousValue: event.previousValue, newValue: event.newValue },\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    };\n    notifier.on('state-change', handler);\n    cleanups.push(() => notifier.removeListener('state-change', handler));\n  } catch { /* StateChangeNotifier not registered */ }\n\n  return cleanups;\n}\n"]}
|
|
335
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LogHooks.js","sourceRoot":"","sources":["../../src/logging/LogHooks.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Phase 4: Integration Hooks for Unified Logging System.\n *\n * Maps native events from 10 monitoring/logging systems into UnifiedLogEntry\n * objects and routes them through LogManager.  Each source system exposes a\n * lightweight `addLogListener` callback; the translation logic lives here so\n * source files stay minimal and never import UnifiedLogEntry.\n *\n * Usage:\n *   const cleanups = wireLogHooks(logManager, container);\n *   // later …\n *   cleanups.forEach(fn => fn());\n */\n\nimport type { LogManager } from './LogManager.js';\nimport type { LogLevel, UnifiedLogEntry } from './types.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { DefaultElementProvider } from '../portfolio/DefaultElementProvider.js';\nimport { LRUCache } from '../cache/LRUCache.js';\nimport { EventDeduplicator } from '../utils/EventDeduplicator.js';\n\n// ---------------------------------------------------------------------------\n// Severity → LogLevel helper (shared by SecurityMonitor, SecurityTelemetry,\n// SecurityAuditor)\n// ---------------------------------------------------------------------------\n\nconst SEVERITY_TO_LEVEL: Record<string, LogLevel> = {\n  CRITICAL: 'error',\n  HIGH: 'error',\n  MEDIUM: 'warn',\n  LOW: 'info',\n  critical: 'error',\n  high: 'error',\n  medium: 'warn',\n  low: 'info',\n};\n\nconst CONTENT_INJECTION_VISIBILITY_WINDOW_MS = 10 * 60_000;\nconst CONTENT_INJECTION_VISIBILITY_MAX_KEYS = 500;\n\nfunction extractContentInjectionKey(value: unknown): string | null {\n  if (typeof value !== 'string') {\n    return null;\n  }\n\n  const normalizedValue = UnicodeValidator.normalize(value).normalizedContent;\n  const normalized = normalizedValue.replace(/^Detected pattern:\\s*/i, '').trim();\n  return normalized === '' ? null : `CONTENT_INJECTION\\0${normalized}`;\n}\n\n// ---------------------------------------------------------------------------\n// CorrelationId provider interface (subset of ContextTracker)\n// ---------------------------------------------------------------------------\n\ntype CorrelationIdProvider = { getCorrelationId(): string | undefined };\n\n// ---------------------------------------------------------------------------\n// Exported factory for TriggerMetricsTracker (created outside DI container)\n// ---------------------------------------------------------------------------\n\nexport function getTriggerMetricsLogListener(\n  logManager: LogManager,\n  contextTracker?: CorrelationIdProvider,\n): (level: 'debug' | 'info' | 'warn' | 'error', message: string, data?: Record<string, unknown>) => void {\n  return (level, message, data) => {\n    const entry: UnifiedLogEntry = {\n      id: logManager.generateId(),\n      timestamp: new Date().toISOString(),\n      category: 'telemetry',\n      level,\n      source: 'TriggerMetricsTracker',\n      message,\n      data,\n      correlationId: contextTracker?.getCorrelationId(),\n    };\n    logManager.log(entry);\n  };\n}\n\n// ---------------------------------------------------------------------------\n// Exported factory for SecurityAuditor (created outside DI container)\n// ---------------------------------------------------------------------------\n\nexport function getSecurityAuditorLogListener(\n  logManager: LogManager,\n  contextTracker?: CorrelationIdProvider,\n): (level: 'debug' | 'info' | 'warn' | 'error', message: string, data?: Record<string, unknown>) => void {\n  return (level, message, data) => {\n    const entry: UnifiedLogEntry = {\n      id: logManager.generateId(),\n      timestamp: new Date().toISOString(),\n      category: 'security',\n      level,\n      source: 'SecurityAuditor',\n      message,\n      data,\n      correlationId: contextTracker?.getCorrelationId(),\n    };\n    logManager.log(entry);\n  };\n}\n\n// ---------------------------------------------------------------------------\n// Main wiring function\n// ---------------------------------------------------------------------------\n\n/**\n * Wire all monitoring systems into the unified logging pipeline.\n *\n * @param logManager  The LogManager singleton from the DI container.\n * @param container   The DI container (used to resolve source services).\n * @returns Array of unsubscribe functions — call them during shutdown.\n */\nexport function wireLogHooks(\n  logManager: LogManager,\n  container: { resolve<T>(name: string): T },\n): (() => void)[] {\n  const cleanups: (() => void)[] = [];\n  const contentInjectionVisibilityDedup = new EventDeduplicator(\n    CONTENT_INJECTION_VISIBILITY_WINDOW_MS,\n    CONTENT_INJECTION_VISIBILITY_MAX_KEYS,\n  );\n\n  // Resolve ContextTracker for correlationId injection\n  let contextTracker: CorrelationIdProvider | null = null;\n  try {\n    contextTracker = container.resolve<CorrelationIdProvider>('ContextTracker');\n  } catch { /* ContextTracker not registered */ }\n\n  // --- MCPLogger (application) -------------------------------------------\n  try {\n    const mcpLogger = container.resolve<{\n      addLogListener(fn: (entry: { timestamp: Date; level: string; message: string; data?: any }) => void): () => void;\n    }>('MCPLogger');\n    const unsub = mcpLogger.addLogListener((logEntry) => {\n      const contentInjectionKey =\n        logEntry.message === '[CRITICAL SECURITY ALERT]' &&\n        logEntry.data?.type === 'CONTENT_INJECTION_ATTEMPT'\n          ? extractContentInjectionKey(logEntry.data?.details)\n          : null;\n      if (contentInjectionKey !== null && contentInjectionVisibilityDedup.shouldSuppress(contentInjectionKey)) {\n        return;\n      }\n\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: logEntry.timestamp.toISOString(),\n        category: 'application',\n        level: logEntry.level as LogLevel,\n        source: 'MCPLogger',\n        message: logEntry.message,\n        data: logEntry.data != null ? logEntry.data : undefined,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* MCPLogger not registered */ }\n\n  // --- SecurityMonitor (security, static) ---------------------------------\n  {\n    const unsub = SecurityMonitor.addLogListener((logEntry) => {\n      const contentInjectionKey =\n        logEntry.type === 'CONTENT_INJECTION_ATTEMPT'\n          ? extractContentInjectionKey(logEntry.details)\n          : null;\n      if (contentInjectionKey !== null && contentInjectionVisibilityDedup.shouldSuppress(contentInjectionKey)) {\n        return;\n      }\n\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: logEntry.timestamp,\n        category: 'security',\n        level: SEVERITY_TO_LEVEL[logEntry.severity] ?? 'info',\n        source: 'SecurityMonitor',\n        message: `[${logEntry.type}] ${logEntry.details}`,\n        data: {\n          ...logEntry.additionalData,\n          eventType: logEntry.type,\n          severity: logEntry.severity,\n          sourceComponent: logEntry.source,\n        },\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  }\n\n  // --- SecurityTelemetry (security) ---------------------------------------\n  try {\n    const secTelemetry = container.resolve<{\n      addLogListener(fn: (entry: {\n        timestamp: string; attackType: string; pattern: string;\n        severity: string; source: string; metadata?: Record<string, any>;\n      }) => void): () => void;\n    }>('SecurityTelemetry');\n    const unsub = secTelemetry.addLogListener((telEntry) => {\n      const contentInjectionKey =\n        telEntry.attackType === 'CONTENT_INJECTION'\n          ? extractContentInjectionKey(telEntry.pattern)\n          : null;\n      if (contentInjectionKey !== null && contentInjectionVisibilityDedup.shouldSuppress(contentInjectionKey)) {\n        return;\n      }\n\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: telEntry.timestamp,\n        category: 'security',\n        level: SEVERITY_TO_LEVEL[telEntry.severity] ?? 'info',\n        source: 'SecurityTelemetry',\n        message: `Blocked ${telEntry.attackType}: ${telEntry.pattern}`,\n        data: telEntry.metadata,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* SecurityTelemetry not registered */ }\n\n  // --- PerformanceMonitor (performance) -----------------------------------\n  try {\n    const perfMonitor = container.resolve<{\n      addLogListener(fn: (level: string, message: string, data?: Record<string, unknown>) => void): () => void;\n    }>('PerformanceMonitor');\n    const unsub = perfMonitor.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'performance',\n        level: level as LogLevel,\n        source: 'PerformanceMonitor',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* PerformanceMonitor not registered */ }\n\n  // --- ElementEventDispatcher (application) — already has on() -----------\n  try {\n    const dispatcher = container.resolve<{\n      on(event: string, handler: (payload: any) => void): () => void;\n    }>('ElementEventDispatcher');\n\n    const eventLevelMap: Record<string, LogLevel> = {\n      'element:load:error': 'error',\n      'element:save:error': 'error',\n      'element:delete:error': 'error',\n      'element:lock-timeout': 'warn',\n      'element:activate': 'info',\n      'element:deactivate': 'info',\n      'element:load:success': 'info',\n      'element:save:success': 'info',\n      'element:delete:success': 'info',\n    };\n\n    // Only log events that have a mapped level (errors, warnings, success, activate/deactivate).\n    // Skip start/cache/external-change events — they fire per-element and create noise\n    // without adding value beyond the completion/error logs.\n    const loggedEvents = Object.keys(eventLevelMap);\n\n    for (const eventName of loggedEvents) {\n      const unsub = dispatcher.on(eventName, (payload: any) => {\n        const level = eventLevelMap[eventName] ?? 'debug';\n        const requestCorrelationId = contextTracker?.getCorrelationId();\n        // Use elementId if available, fall back to filename from filePath\n        const elementName = payload.elementId\n          || (payload.filePath ? payload.filePath.replace(/\\.[^.]+$/, '') : '');\n        const entry: UnifiedLogEntry = {\n          id: logManager.generateId(),\n          timestamp: new Date().toISOString(),\n          category: 'application',\n          level,\n          source: 'ElementEventDispatcher',\n          message: `${eventName} [${payload.elementType ?? 'unknown'}:${elementName}]`,\n          data: {\n            ...payload.extra,\n            ...(payload.correlationId ? { operationId: payload.correlationId } : {}),\n            ...(payload.filePath ? { filePath: payload.filePath } : {}),\n          },\n          correlationId: requestCorrelationId ?? payload.correlationId,\n        };\n        logManager.log(entry);\n      });\n      cleanups.push(unsub);\n    }\n  } catch { /* ElementEventDispatcher not registered */ }\n\n  // --- OperationalTelemetry (telemetry) -----------------------------------\n  try {\n    const opsTelemetry = container.resolve<{\n      addLogListener(fn: (level: string, message: string, data?: Record<string, unknown>) => void): () => void;\n    }>('OperationalTelemetry');\n    const unsub = opsTelemetry.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'telemetry',\n        level: level as LogLevel,\n        source: 'OperationalTelemetry',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* OperationalTelemetry not registered */ }\n\n  // --- FileLockManager (performance) --------------------------------------\n  try {\n    const lockManager = container.resolve<{\n      addLogListener(fn: (level: string, message: string, data?: Record<string, unknown>) => void): () => void;\n    }>('FileLockManager');\n    const unsub = lockManager.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'performance',\n        level: level as LogLevel,\n        source: 'FileLockManager',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  } catch { /* FileLockManager not registered */ }\n\n  // --- DefaultElementProvider (performance, static) -----------------------\n  {\n    const unsub = DefaultElementProvider.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'performance',\n        level: level as LogLevel,\n        source: 'DefaultElementProvider',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  }\n\n  // --- LRUCache (performance, static) ------------------------------------\n  {\n    const unsub = LRUCache.addLogListener((level, message, data) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'performance',\n        level: level as LogLevel,\n        source: 'LRUCache',\n        message,\n        data,\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    });\n    cleanups.push(unsub);\n  }\n\n  // --- StateChangeNotifier (application) — extends EventEmitter ----------\n  try {\n    const notifier = container.resolve<{\n      on(event: string, handler: (...args: any[]) => void): any;\n      removeListener(event: string, handler: (...args: any[]) => void): any;\n    }>('StateChangeNotifier');\n    const handler = (event: { type: string; previousValue: string | null; newValue: string | null }) => {\n      const entry: UnifiedLogEntry = {\n        id: logManager.generateId(),\n        timestamp: new Date().toISOString(),\n        category: 'application',\n        level: 'info',\n        source: 'StateChangeNotifier',\n        message: `State change: ${event.type}`,\n        data: { previousValue: event.previousValue, newValue: event.newValue },\n        correlationId: contextTracker?.getCorrelationId(),\n      };\n      logManager.log(entry);\n    };\n    notifier.on('state-change', handler);\n    cleanups.push(() => notifier.removeListener('state-change', handler));\n  } catch { /* StateChangeNotifier not registered */ }\n\n  return cleanups;\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"contentPipeline.d.ts","sourceRoot":"","sources":["../../src/web/contentPipeline.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;
|
|
1
|
+
{"version":3,"file":"contentPipeline.d.ts","sourceRoot":"","sources":["../../src/web/contentPipeline.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAmBH,4EAA4E;AAC5E,MAAM,WAAW,sBAAsB;IACrC,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;IACpB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,qBAAqB,CAAC,EAAE,MAAM,CAAC;IAC/B,SAAS,CAAC,EAAE,MAAM,EAAE,CAAC;IACrB,kBAAkB,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAC5C,UAAU,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACrC,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC/B,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACnC,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED,MAAM,WAAW,cAAc;IAC7B,KAAK,EAAE,OAAO,CAAC;IACf,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,sBAAsB,CAAC;IACjC,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,CAAC,EAAE;QACV,MAAM,EAAE,MAAM,CAAC;QACf,QAAQ,CAAC,EAAE,MAAM,CAAC;QAClB,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;KACrB,CAAC;CACH;AA+BD,wBAAgB,mCAAmC,IAAI,IAAI,CAE1D;AAED;;;;;;;GAOG;AACH,wBAAgB,sBAAsB,CACpC,QAAQ,EAAE,MAAM,EAChB,UAAU,EAAE,MAAM,EAClB,WAAW,EAAE,MAAM,GAClB,cAAc,CA8FhB"}
|
|
@@ -11,10 +11,12 @@
|
|
|
11
11
|
* auto-dollhouse#5 / upstream #1679
|
|
12
12
|
* DMCP-SEC-004 compliant: uses UnicodeValidator.normalize() on all inputs
|
|
13
13
|
*/
|
|
14
|
+
import { createHash } from 'node:crypto';
|
|
14
15
|
import { extname } from 'node:path';
|
|
15
16
|
import { SecureYamlParser } from '../security/secureYamlParser.js';
|
|
16
17
|
import { ContentValidator } from '../security/contentValidator.js';
|
|
17
18
|
import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
|
|
19
|
+
import { LRUCache } from '../cache/LRUCache.js';
|
|
18
20
|
import { logger } from '../utils/logger.js';
|
|
19
21
|
/** Element types that map to content validation contexts */
|
|
20
22
|
const TYPE_TO_CONTEXT = {
|
|
@@ -24,6 +26,34 @@ const TYPE_TO_CONTEXT = {
|
|
|
24
26
|
agents: 'agent',
|
|
25
27
|
memories: 'memory',
|
|
26
28
|
};
|
|
29
|
+
const CONTENT_PIPELINE_CACHE_MAX_SIZE = 256;
|
|
30
|
+
const CONTENT_PIPELINE_CACHE_MAX_MEMORY_MB = 16;
|
|
31
|
+
const CONTENT_PIPELINE_CACHE = new LRUCache({
|
|
32
|
+
name: 'web-content-validation',
|
|
33
|
+
maxSize: CONTENT_PIPELINE_CACHE_MAX_SIZE,
|
|
34
|
+
maxMemoryMB: CONTENT_PIPELINE_CACHE_MAX_MEMORY_MB,
|
|
35
|
+
});
|
|
36
|
+
function buildContentCacheKey(filename, elementType, rawContent) {
|
|
37
|
+
const contentHash = createHash('sha256').update(rawContent).digest('hex');
|
|
38
|
+
// Keep filename/type in the key so identical payloads from different routes
|
|
39
|
+
// remain independently attributable if route-specific handling diverges later.
|
|
40
|
+
return JSON.stringify([elementType, filename, contentHash]);
|
|
41
|
+
}
|
|
42
|
+
function clonePipelineResult(result) {
|
|
43
|
+
return {
|
|
44
|
+
...result,
|
|
45
|
+
metadata: { ...result.metadata },
|
|
46
|
+
rejection: result.rejection
|
|
47
|
+
? {
|
|
48
|
+
...result.rejection,
|
|
49
|
+
patterns: result.rejection.patterns ? [...result.rejection.patterns] : undefined,
|
|
50
|
+
}
|
|
51
|
+
: undefined,
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
export function resetContentPipelineCacheForTesting() {
|
|
55
|
+
CONTENT_PIPELINE_CACHE.clear();
|
|
56
|
+
}
|
|
27
57
|
/**
|
|
28
58
|
* Validate element content through the security pipeline.
|
|
29
59
|
*
|
|
@@ -40,6 +70,11 @@ export function validateElementContent(filename, rawContent, elementType) {
|
|
|
40
70
|
const normalizedFilename = filenameValidation.normalizedContent;
|
|
41
71
|
const normalizedContent = contentValidation.normalizedContent;
|
|
42
72
|
const normalizedType = elementType.normalize('NFC');
|
|
73
|
+
const cacheKey = buildContentCacheKey(normalizedFilename, normalizedType, rawContent);
|
|
74
|
+
const cachedResult = CONTENT_PIPELINE_CACHE.get(cacheKey);
|
|
75
|
+
if (cachedResult) {
|
|
76
|
+
return clonePipelineResult(cachedResult);
|
|
77
|
+
}
|
|
43
78
|
const ext = extname(normalizedFilename);
|
|
44
79
|
const isYaml = ext === '.yaml' || ext === '.yml';
|
|
45
80
|
const contentContext = TYPE_TO_CONTEXT[normalizedType];
|
|
@@ -65,13 +100,18 @@ export function validateElementContent(filename, rawContent, elementType) {
|
|
|
65
100
|
}
|
|
66
101
|
}
|
|
67
102
|
catch (err) {
|
|
68
|
-
|
|
103
|
+
const result = {
|
|
69
104
|
valid: false,
|
|
70
105
|
content: normalizedContent,
|
|
71
106
|
metadata: {},
|
|
72
107
|
body: '',
|
|
73
108
|
rejection: { reason: `Parse validation failed: ${err.message}`, severity: 'high' },
|
|
74
109
|
};
|
|
110
|
+
// Cache parse failures too: steady-state polling can otherwise keep reparsing
|
|
111
|
+
// identical malformed files forever. If the file is fixed, the content hash
|
|
112
|
+
// changes and this entry is naturally bypassed.
|
|
113
|
+
CONTENT_PIPELINE_CACHE.set(cacheKey, clonePipelineResult(result));
|
|
114
|
+
return result;
|
|
75
115
|
}
|
|
76
116
|
// Step 2: Content injection pattern detection (markdown elements only)
|
|
77
117
|
// YAML memories skip this — they contain legitimate code patterns and
|
|
@@ -88,7 +128,7 @@ export function validateElementContent(filename, rawContent, elementType) {
|
|
|
88
128
|
patterns: validation.detectedPatterns,
|
|
89
129
|
severity: validation.severity,
|
|
90
130
|
});
|
|
91
|
-
|
|
131
|
+
const result = {
|
|
92
132
|
valid: false,
|
|
93
133
|
content: normalizedContent,
|
|
94
134
|
metadata,
|
|
@@ -99,14 +139,18 @@ export function validateElementContent(filename, rawContent, elementType) {
|
|
|
99
139
|
patterns: validation.detectedPatterns,
|
|
100
140
|
},
|
|
101
141
|
};
|
|
142
|
+
CONTENT_PIPELINE_CACHE.set(cacheKey, clonePipelineResult(result));
|
|
143
|
+
return result;
|
|
102
144
|
}
|
|
103
145
|
}
|
|
104
146
|
// Step 3: Return validated content
|
|
105
|
-
|
|
147
|
+
const result = {
|
|
106
148
|
valid: true,
|
|
107
149
|
content: rawContent,
|
|
108
150
|
metadata,
|
|
109
151
|
body,
|
|
110
152
|
};
|
|
153
|
+
CONTENT_PIPELINE_CACHE.set(cacheKey, clonePipelineResult(result));
|
|
154
|
+
return result;
|
|
111
155
|
}
|
|
112
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"contentPipeline.js","sourceRoot":"","sources":["../../src/web/contentPipeline.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Content Validation Pipeline for Web UI\n *\n * Composes the standalone security validators into a single pipeline\n * for validating portfolio element content before serving to the browser.\n *\n * Uses the same validators as the MCPAQLHandler but without coupling\n * to the handler's operation dispatch. This is the extraction point\n * for the upcoming MCPAQLHandler decomposition.\n *\n * auto-dollhouse#5 / upstream #1679\n * DMCP-SEC-004 compliant: uses UnicodeValidator.normalize() on all inputs\n */\n\nimport { extname } from 'node:path';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\nimport { ContentValidator, type ContentValidationResult } from '../security/contentValidator.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { logger } from '../utils/logger.js';\n\n/** Element types that map to content validation contexts */\nconst TYPE_TO_CONTEXT: Record<string, 'persona' | 'skill' | 'template' | 'agent' | 'memory'> = {\n  personas: 'persona',\n  skills: 'skill',\n  templates: 'template',\n  agents: 'agent',\n  memories: 'memory',\n};\n\n/** Known metadata fields extracted from YAML frontmatter for web display */\nexport interface ElementDisplayMetadata {\n  name?: string;\n  description?: string;\n  version?: string;\n  author?: string;\n  category?: string;\n  created?: string;\n  created_date?: string;\n  modified?: string;\n  tags?: string[];\n  license?: string;\n  age_rating?: string;\n  triggers?: string[];\n  instructions?: string;\n  coordination_strategy?: string;\n  use_cases?: string[];\n  proficiency_levels?: Record<string, string>;\n  gatekeeper?: Record<string, unknown>;\n  goal?: Record<string, unknown>;\n  autonomy?: Record<string, unknown>;\n  memoryType?: string;\n  [key: string]: unknown;\n}\n\nexport interface PipelineResult {\n  valid: boolean;\n  content: string;\n  metadata: ElementDisplayMetadata;\n  body: string;\n  rejection?: {\n    reason: string;\n    severity?: string;\n    patterns?: string[];\n  };\n}\n\n/**\n * Validate element content through the security pipeline.\n *\n * @param filename - The element filename (e.g., \"alex-sterling.md\")\n * @param rawContent - The raw file content string\n * @param elementType - The plural element type directory (e.g., \"personas\")\n * @returns Validated content with parsed metadata, or rejection with reason\n */\nexport function validateElementContent(\n  filename: string,\n  rawContent: string,\n  elementType: string,\n): PipelineResult {\n  // DMCP-SEC-004: Normalize all inputs via UnicodeValidator to prevent\n  // homograph attacks, direction override bypasses, and suspicious patterns.\n  const filenameValidation = UnicodeValidator.normalize(filename);\n  const contentValidation = UnicodeValidator.normalize(rawContent);\n  const normalizedFilename = filenameValidation.normalizedContent;\n  const normalizedContent = contentValidation.normalizedContent;\n  const normalizedType = elementType.normalize('NFC');\n\n  const ext = extname(normalizedFilename);\n  const isYaml = ext === '.yaml' || ext === '.yml';\n  const contentContext = TYPE_TO_CONTEXT[normalizedType];\n\n  // Step 1: Parse and validate structure (YAML bomb detection, circular refs)\n  let metadata: ElementDisplayMetadata = {};\n  let body = '';\n\n  try {\n    if (isYaml) {\n      // Pure YAML file (memories) — use parseRawYaml for structural validation\n      // (bomb detection, circular refs, size limits).\n      // Skip ContentValidator.validateYamlContent() — it produces false positives\n      // on memory content that legitimately contains code patterns, security\n      // keywords, and technical documentation. Memories are locally-generated\n      // trusted content, not untrusted external submissions.\n      const parsed = SecureYamlParser.parseRawYaml(normalizedContent);\n      metadata = (typeof parsed === 'object' && parsed !== null) ? parsed : {};\n    } else {\n      // Markdown with YAML frontmatter\n      const parsed = SecureYamlParser.parse(normalizedContent, { contentContext });\n      metadata = parsed.data;\n      body = parsed.content;\n    }\n  } catch (err) {\n    return {\n      valid: false,\n      content: normalizedContent,\n      metadata: {},\n      body: '',\n      rejection: { reason: `Parse validation failed: ${(err as Error).message}`, severity: 'high' },\n    };\n  }\n\n  // Step 2: Content injection pattern detection (markdown elements only)\n  // YAML memories skip this — they contain legitimate code patterns and\n  // technical content that triggers false positives. The structural YAML\n  // parsing above (bomb/circular ref detection) is sufficient for local content.\n  if (!isYaml) {\n    const validation: ContentValidationResult = ContentValidator.validateAndSanitize(normalizedContent, {\n      contentContext,\n    });\n\n    if (!validation.isValid) {\n      logger.warn('[ContentPipeline] Content rejected', {\n        filename,\n        elementType,\n        patterns: validation.detectedPatterns,\n        severity: validation.severity,\n      });\n      return {\n        valid: false,\n        content: normalizedContent,\n        metadata,\n        body,\n        rejection: {\n          reason: 'Content failed security validation',\n          severity: validation.severity,\n          patterns: validation.detectedPatterns,\n        },\n      };\n    }\n  }\n\n  // Step 3: Return validated content\n  return {\n    valid: true,\n    content: rawContent,\n    metadata,\n    body,\n  };\n}\n"]}
|
|
156
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"contentPipeline.js","sourceRoot":"","sources":["../../src/web/contentPipeline.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Content Validation Pipeline for Web UI\n *\n * Composes the standalone security validators into a single pipeline\n * for validating portfolio element content before serving to the browser.\n *\n * Uses the same validators as the MCPAQLHandler but without coupling\n * to the handler's operation dispatch. This is the extraction point\n * for the upcoming MCPAQLHandler decomposition.\n *\n * auto-dollhouse#5 / upstream #1679\n * DMCP-SEC-004 compliant: uses UnicodeValidator.normalize() on all inputs\n */\n\nimport { createHash } from 'node:crypto';\nimport { extname } from 'node:path';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\nimport { ContentValidator, type ContentValidationResult } from '../security/contentValidator.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { LRUCache } from '../cache/LRUCache.js';\nimport { logger } from '../utils/logger.js';\n\n/** Element types that map to content validation contexts */\nconst TYPE_TO_CONTEXT: Record<string, 'persona' | 'skill' | 'template' | 'agent' | 'memory'> = {\n  personas: 'persona',\n  skills: 'skill',\n  templates: 'template',\n  agents: 'agent',\n  memories: 'memory',\n};\n\n/** Known metadata fields extracted from YAML frontmatter for web display */\nexport interface ElementDisplayMetadata {\n  name?: string;\n  description?: string;\n  version?: string;\n  author?: string;\n  category?: string;\n  created?: string;\n  created_date?: string;\n  modified?: string;\n  tags?: string[];\n  license?: string;\n  age_rating?: string;\n  triggers?: string[];\n  instructions?: string;\n  coordination_strategy?: string;\n  use_cases?: string[];\n  proficiency_levels?: Record<string, string>;\n  gatekeeper?: Record<string, unknown>;\n  goal?: Record<string, unknown>;\n  autonomy?: Record<string, unknown>;\n  memoryType?: string;\n  [key: string]: unknown;\n}\n\nexport interface PipelineResult {\n  valid: boolean;\n  content: string;\n  metadata: ElementDisplayMetadata;\n  body: string;\n  rejection?: {\n    reason: string;\n    severity?: string;\n    patterns?: string[];\n  };\n}\n\nconst CONTENT_PIPELINE_CACHE_MAX_SIZE = 256;\nconst CONTENT_PIPELINE_CACHE_MAX_MEMORY_MB = 16;\n\nconst CONTENT_PIPELINE_CACHE = new LRUCache<PipelineResult>({\n  name: 'web-content-validation',\n  maxSize: CONTENT_PIPELINE_CACHE_MAX_SIZE,\n  maxMemoryMB: CONTENT_PIPELINE_CACHE_MAX_MEMORY_MB,\n});\n\nfunction buildContentCacheKey(filename: string, elementType: string, rawContent: string): string {\n  const contentHash = createHash('sha256').update(rawContent).digest('hex');\n  // Keep filename/type in the key so identical payloads from different routes\n  // remain independently attributable if route-specific handling diverges later.\n  return JSON.stringify([elementType, filename, contentHash]);\n}\n\nfunction clonePipelineResult(result: PipelineResult): PipelineResult {\n  return {\n    ...result,\n    metadata: { ...result.metadata },\n    rejection: result.rejection\n      ? {\n          ...result.rejection,\n          patterns: result.rejection.patterns ? [...result.rejection.patterns] : undefined,\n        }\n      : undefined,\n  };\n}\n\nexport function resetContentPipelineCacheForTesting(): void {\n  CONTENT_PIPELINE_CACHE.clear();\n}\n\n/**\n * Validate element content through the security pipeline.\n *\n * @param filename - The element filename (e.g., \"alex-sterling.md\")\n * @param rawContent - The raw file content string\n * @param elementType - The plural element type directory (e.g., \"personas\")\n * @returns Validated content with parsed metadata, or rejection with reason\n */\nexport function validateElementContent(\n  filename: string,\n  rawContent: string,\n  elementType: string,\n): PipelineResult {\n  // DMCP-SEC-004: Normalize all inputs via UnicodeValidator to prevent\n  // homograph attacks, direction override bypasses, and suspicious patterns.\n  const filenameValidation = UnicodeValidator.normalize(filename);\n  const contentValidation = UnicodeValidator.normalize(rawContent);\n  const normalizedFilename = filenameValidation.normalizedContent;\n  const normalizedContent = contentValidation.normalizedContent;\n  const normalizedType = elementType.normalize('NFC');\n  const cacheKey = buildContentCacheKey(normalizedFilename, normalizedType, rawContent);\n  const cachedResult = CONTENT_PIPELINE_CACHE.get(cacheKey);\n  if (cachedResult) {\n    return clonePipelineResult(cachedResult);\n  }\n\n  const ext = extname(normalizedFilename);\n  const isYaml = ext === '.yaml' || ext === '.yml';\n  const contentContext = TYPE_TO_CONTEXT[normalizedType];\n\n  // Step 1: Parse and validate structure (YAML bomb detection, circular refs)\n  let metadata: ElementDisplayMetadata = {};\n  let body = '';\n\n  try {\n    if (isYaml) {\n      // Pure YAML file (memories) — use parseRawYaml for structural validation\n      // (bomb detection, circular refs, size limits).\n      // Skip ContentValidator.validateYamlContent() — it produces false positives\n      // on memory content that legitimately contains code patterns, security\n      // keywords, and technical documentation. Memories are locally-generated\n      // trusted content, not untrusted external submissions.\n      const parsed = SecureYamlParser.parseRawYaml(normalizedContent);\n      metadata = (typeof parsed === 'object' && parsed !== null) ? parsed : {};\n    } else {\n      // Markdown with YAML frontmatter\n      const parsed = SecureYamlParser.parse(normalizedContent, { contentContext });\n      metadata = parsed.data;\n      body = parsed.content;\n    }\n  } catch (err) {\n    const result: PipelineResult = {\n      valid: false,\n      content: normalizedContent,\n      metadata: {},\n      body: '',\n      rejection: { reason: `Parse validation failed: ${(err as Error).message}`, severity: 'high' },\n    };\n    // Cache parse failures too: steady-state polling can otherwise keep reparsing\n    // identical malformed files forever. If the file is fixed, the content hash\n    // changes and this entry is naturally bypassed.\n    CONTENT_PIPELINE_CACHE.set(cacheKey, clonePipelineResult(result));\n    return result;\n  }\n\n  // Step 2: Content injection pattern detection (markdown elements only)\n  // YAML memories skip this — they contain legitimate code patterns and\n  // technical content that triggers false positives. The structural YAML\n  // parsing above (bomb/circular ref detection) is sufficient for local content.\n  if (!isYaml) {\n    const validation: ContentValidationResult = ContentValidator.validateAndSanitize(normalizedContent, {\n      contentContext,\n    });\n\n    if (!validation.isValid) {\n      logger.warn('[ContentPipeline] Content rejected', {\n        filename,\n        elementType,\n        patterns: validation.detectedPatterns,\n        severity: validation.severity,\n      });\n      const result: PipelineResult = {\n        valid: false,\n        content: normalizedContent,\n        metadata,\n        body,\n        rejection: {\n          reason: 'Content failed security validation',\n          severity: validation.severity,\n          patterns: validation.detectedPatterns,\n        },\n      };\n      CONTENT_PIPELINE_CACHE.set(cacheKey, clonePipelineResult(result));\n      return result;\n    }\n  }\n\n  // Step 3: Return validated content\n  const result: PipelineResult = {\n    valid: true,\n    content: rawContent,\n    metadata,\n    body,\n  };\n  CONTENT_PIPELINE_CACHE.set(cacheKey, clonePipelineResult(result));\n  return result;\n}\n"]}
|