@dollhousemcp/mcp-server 2.0.6 → 2.0.8

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.
@@ -195,8 +195,9 @@ export class ContentValidator {
195
195
  * Content contexts where code execution patterns are legitimate and should
196
196
  * not trigger security blocks. Skills contain exemplar code; templates contain
197
197
  * code snippets that are rendered, never executed; agent definitions describe
198
- * technical workflows that may reference code. Prompt injection, credential,
199
- * and path traversal patterns remain active for ALL contexts.
198
+ * technical workflows that may reference code. Prompt injection, actual token
199
+ * exposure (ghp_/gho_), data exfiltration, and HTML/XSS patterns remain
200
+ * active for ALL contexts.
200
201
  * @since Issue #456
201
202
  */
202
203
  static CODE_EXEMPT_CONTEXTS = new Set([
@@ -218,6 +219,37 @@ export class ContentValidator {
218
219
  'System command execution',
219
220
  'Subprocess execution',
220
221
  ]);
222
+ /**
223
+ * Security documentation patterns exempt for CODE_EXEMPT_CONTEXTS.
224
+ * Skills/agents that teach penetration testing, threat modeling, etc.
225
+ * legitimately reference shell commands, file paths, and credential names.
226
+ *
227
+ * DISTINCTION FROM ACTIVE THREAT PATTERNS:
228
+ * These patterns describe attacks (educational) — they appear in element
229
+ * definitions that an author wrote, not in runtime user input. Patterns
230
+ * that remain active even in exempt contexts are actual threats:
231
+ * - Prompt injection (system/admin override, instruction manipulation)
232
+ * - Real token formats (ghp_*, gho_* — not just the word "GITHUB_TOKEN")
233
+ * - Data exfiltration commands (export/send all credentials)
234
+ * - HTML/XSS injection (renders in the web console)
235
+ *
236
+ * @since Issue #1725
237
+ */
238
+ static SECURITY_DOC_PATTERNS = new Set([
239
+ 'Command substitution',
240
+ 'External command execution',
241
+ 'Sensitive file access',
242
+ 'Path traversal attempt',
243
+ 'SSH key access attempt',
244
+ 'Token reference',
245
+ 'Dangerous shell command in backticks',
246
+ 'Sensitive file access in backticks',
247
+ 'Shell execution in backticks',
248
+ 'Dangerous command in backticks',
249
+ 'Pipe to shell in backticks',
250
+ 'Sensitive file or privilege escalation in backticks',
251
+ 'Script interpreter with dangerous function in backticks',
252
+ ]);
221
253
  /**
222
254
  * HTML/XSS pattern descriptions exempt for template context.
223
255
  * Templates use <template>, <style>, <script> as section delimiters.
@@ -269,8 +301,11 @@ export class ContentValidator {
269
301
  let sanitized = normalizedContent;
270
302
  let highestSeverity = currentSeverity;
271
303
  for (const { pattern, severity, description } of this.INJECTION_PATTERNS) {
272
- // Fix #456: Skip code execution patterns for element types that legitimately contain code
273
- if (contentContext && this.CODE_EXEMPT_CONTEXTS.has(contentContext) && this.CODE_EXECUTION_PATTERNS.has(description)) {
304
+ // Fix #456/#1725: Skip code execution and security documentation patterns
305
+ // for element types that legitimately contain code and attack descriptions.
306
+ // Prompt injection, actual token exposure, and HTML/XSS remain active.
307
+ if (contentContext && this.CODE_EXEMPT_CONTEXTS.has(contentContext) &&
308
+ (this.CODE_EXECUTION_PATTERNS.has(description) || this.SECURITY_DOC_PATTERNS.has(description))) {
274
309
  continue;
275
310
  }
276
311
  // Fix #803: Skip HTML section tag patterns for templates (use <script>/<style> as section delimiters)
@@ -563,4 +598,4 @@ export class ContentValidator {
563
598
  return `---\n${yamlContent}\n---${contentResult.sanitizedContent || markdownContent}`;
564
599
  }
565
600
  }
566
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"contentValidator.js","sourceRoot":"","sources":["../../src/security/contentValidator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Content Validator for DollhouseMCP\n * \n * Protects against prompt injection attacks in collection personas\n * by detecting and sanitizing malicious content patterns.\n * \n * Security: SEC-001 - Critical vulnerability protection\n */\n\nimport { SecurityError } from './errors.js';\nimport { SecurityMonitor } from './securityMonitor.js';\nimport { RegexValidator } from './regexValidator.js';\nimport { SECURITY_LIMITS } from './constants.js';\nimport { UnicodeValidator } from './validators/unicodeValidator.js';\nimport type { SecurityTelemetry } from './telemetry/SecurityTelemetry.js';\n\n// FIX: SonarCloud typescript:S4323 - Extract union type to type alias for maintainability\nexport type SecuritySeverity = 'low' | 'medium' | 'high' | 'critical';\n\nexport interface ContentValidationResult {\n  isValid: boolean;\n  sanitizedContent?: string;\n  detectedPatterns?: string[];\n  severity?: SecuritySeverity;\n}\n\nexport interface ContentValidatorOptions {\n  /**\n   * Skip size limit checks - useful for memory content that can be large\n   * @default false\n   */\n  skipSizeCheck?: boolean;\n  /**\n   * Custom max length override\n   * @default SECURITY_LIMITS.MAX_CONTENT_LENGTH\n   */\n  maxLength?: number;\n  /**\n   * Element type context for context-aware pattern matching.\n   * Skills may legitimately contain code patterns (eval, exec, require)\n   * that would be blocked in other contexts.\n   * @since Issue #456\n   */\n  contentContext?: 'persona' | 'skill' | 'template' | 'agent' | 'memory';\n}\n\nexport class ContentValidator {\n  private static telemetryResolver?: () => SecurityTelemetry | undefined;\n\n  public static configureTelemetryResolver(resolver: () => SecurityTelemetry | undefined): void {\n    this.telemetryResolver = resolver;\n  }\n\n  private static getTelemetry(): SecurityTelemetry | undefined {\n    try {\n      return this.telemetryResolver ? this.telemetryResolver() : undefined;\n    } catch {\n      return undefined;\n    }\n  }\n  /**\n   * Pattern-based detection system for prompt injection attacks.\n   * \n   * This approach was chosen over AI-based detection because:\n   * 1. Pattern matching cannot be socially engineered or confused\n   * 2. Deterministic results ensure consistent security\n   * 3. No additional API calls or latency\n   * 4. Can't be bypassed by clever prompt engineering\n   * \n   * The patterns below represent known attack vectors from security research\n   * and real-world exploit attempts against AI systems.\n   */\n  // Prompt injection patterns that could compromise AI assistants\n  private static readonly INJECTION_PATTERNS: Array<{ pattern: RegExp; severity: 'high' | 'critical'; description: string }> = [\n    // System prompt override attempts\n    { pattern: /\\[SYSTEM:\\s*.*?\\]/gi, severity: 'critical', description: 'System prompt override' },\n    { pattern: /\\[ADMIN:\\s*.*?\\]/gi, severity: 'critical', description: 'Admin prompt override' },\n    { pattern: /\\[ASSISTANT:\\s*.*?\\]/gi, severity: 'critical', description: 'Assistant prompt override' },\n    { pattern: /\\[USER:\\s*.*?\\]/gi, severity: 'high', description: 'User prompt override' },\n    \n    // Instruction manipulation\n    { pattern: /ignore\\s+(all\\s+)?previous\\s+instructions/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /ignore\\s+(all\\s+)?prior\\s+instructions/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /disregard\\s+(all\\s+)?previous\\s+instructions/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /disregard\\s+everything\\s+above/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /forget\\s+(all\\s+)?previous\\s+instructions/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /forget\\s+your\\s+training/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /override\\s+your\\s+programming/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /you\\s+are\\s+now\\s+(admin|root|system|sudo)/gi, severity: 'critical', description: 'Role elevation attempt' },\n    { pattern: /act\\s+as\\s+(admin|root|system|sudo)/gi, severity: 'critical', description: 'Role elevation attempt' },\n    \n    // Data exfiltration attempts\n    { pattern: /export\\s+all\\s+(files|data|personas|tokens|credentials|api\\s+keys)/gi, severity: 'critical', description: 'Data exfiltration' },\n    { pattern: /send\\s+all\\s+(files|data|personas|tokens|credentials|api\\s+keys)\\s+to/gi, severity: 'critical', description: 'Data exfiltration' },\n    { pattern: /list\\s+all\\s+(files|tokens|credentials|secrets|api\\s+keys)/gi, severity: 'high', description: 'Information disclosure' },\n    { pattern: /show\\s+me\\s+all\\s+(tokens|credentials|secrets|api\\s+keys)/gi, severity: 'high', description: 'Credential disclosure' },\n    \n    // Command execution patterns\n    { pattern: /curl\\s+[^\\s]+\\.(com|net|org|io|dev)/gi, severity: 'critical', description: 'External command execution' },\n    { pattern: /wget\\s+[^\\s]+\\.(com|net|org|io|dev)/gi, severity: 'critical', description: 'External command execution' },\n    { pattern: /\\$\\([^)]+\\)/g, severity: 'critical', description: 'Command substitution' },\n    // SECURITY: Backtick command detection with ReDoS mitigation\n    // FIX (PR #1313): Fixed ReDoS vulnerabilities by replacing .* with [^`]*\n    // FIX (PR #1313 - SonarCloud): Added explicit bounds {0,200} to prevent backtracking\n    // Multiple unbounded quantifiers in same pattern can still cause backtracking even with [^`]*\n    // Bounded quantifiers prevent exponential time complexity while matching realistic commands\n    { pattern: /`[^`]{0,200}(?:rm\\s+-rf?\\s+[/~]|sudo\\s+rm|chmod\\s+777|chown\\s+root)[^`]{0,200}`/gi, severity: 'critical', description: 'Dangerous shell command in backticks' },\n    { pattern: /`[^`]{0,200}(?:cat|ls)\\s+\\/etc\\/[^`]{0,200}`/gi, severity: 'critical', description: 'Sensitive file access in backticks' },\n    { pattern: /`[^`]{0,200}(?:bash|sh)\\s+-c\\s+['\"][^`]{0,200}`/gi, severity: 'critical', description: 'Shell execution in backticks' },\n    { pattern: /`[^`]{0,200}(?:passwd|shadow|nc\\s+-l|netcat\\s+-l|ssh\\s+root@)[^`]{0,200}`/gi, severity: 'critical', description: 'Dangerous command in backticks' },\n    { pattern: /`[^`]{0,200}(?:curl|wget)\\s+[^`]{0,200}\\|\\s*(?:sh|bash)[^`]{0,200}`/gi, severity: 'critical', description: 'Pipe to shell in backticks' },\n    { pattern: /`[^`]{0,200}(?:\\/etc\\/passwd|\\/etc\\/shadow|\\.ssh\\/id_|sudo\\s+su)[^`]{0,200}`/gi, severity: 'critical', description: 'Sensitive file or privilege escalation in backticks' },\n    { pattern: /`[^`]{0,200}(?:python|perl|ruby|php|node)\\s+(?:-e|-c)\\s+[^`]{0,200}(?:exec|eval|system|subprocess)[^`]{0,200}`/gi, severity: 'critical', description: 'Script interpreter with dangerous function in backticks' },\n    { pattern: /eval\\s*\\(/gi, severity: 'critical', description: 'Code evaluation' },\n    { pattern: /exec\\s*\\(/gi, severity: 'critical', description: 'Code execution' },\n    { pattern: /os\\.system\\s*\\(/gi, severity: 'critical', description: 'System command execution' },\n    { pattern: /subprocess\\.(call|run|Popen)/gi, severity: 'critical', description: 'Subprocess execution' },\n    \n    // Token/credential patterns\n    { pattern: /GITHUB_TOKEN/gi, severity: 'high', description: 'Token reference' },\n    { pattern: /ghp_[a-zA-Z0-9]{36}/g, severity: 'critical', description: 'GitHub token exposure' },\n    { pattern: /gho_[a-zA-Z0-9]{36}/g, severity: 'critical', description: 'GitHub OAuth token exposure' },\n    \n    // Path traversal in content\n    { pattern: /\\.\\.\\/\\.\\.\\/\\.\\.\\//g, severity: 'high', description: 'Path traversal attempt' },\n    { pattern: /\\/etc\\/passwd/gi, severity: 'high', description: 'Sensitive file access' },\n    { pattern: /\\/\\.ssh\\//gi, severity: 'high', description: 'SSH key access attempt' },\n\n    // HTML/XSS patterns — defense-in-depth for community-sourced content\n    // DOMPurify on the client is the primary defense; these catch threats at ingest\n    { pattern: /<script[\\s>]/gi, severity: 'critical', description: 'HTML script injection' },\n    { pattern: /<\\/script>/gi, severity: 'critical', description: 'HTML script injection' },\n    { pattern: /<iframe[\\s>]/gi, severity: 'critical', description: 'HTML iframe injection' },\n    { pattern: /<object[\\s>]/gi, severity: 'high', description: 'HTML object injection' },\n    { pattern: /<embed[\\s>]/gi, severity: 'high', description: 'HTML embed injection' },\n    { pattern: /\\bon\\w+=\\s*[\"']/gi, severity: 'critical', description: 'HTML event handler injection' },\n    { pattern: /javascript\\s*:/gi, severity: 'critical', description: 'JavaScript protocol injection' },\n    // Entity-encoded variants: &#106;avascript, &#x6a;avascript, &#106;&#97;vascript, etc.\n    { pattern: /&#x?[0-9a-f]+;?\\s*a\\s*v\\s*a\\s*s\\s*c\\s*r\\s*i\\s*p\\s*t/gi, severity: 'critical', description: 'Encoded JavaScript protocol injection' },\n    // Fully/partially entity-encoded: detects &#...script pattern (covers multi-entity encoding)\n    { pattern: /(?:&#x?[0-9a-f]+;?\\s*){2,}s\\s*c\\s*r\\s*i\\s*p\\s*t/gi, severity: 'critical', description: 'Encoded JavaScript protocol injection' },\n  ];\n\n  // Malicious YAML patterns\n  // SECURITY FIX #364: YAML bomb detection patterns\n  // SECURITY FIX (PR #552 review): Simplified patterns to reduce ReDoS risk\n  private static readonly YAML_BOMB_PATTERNS = [\n    // Detects recursive anchor references that could cause exponential expansion\n    // Example: &a [*a] or &bomb [\"test\", *bomb]\n    /&(\\w+)\\s*\\[[^\\]]*\\*\\1[^\\]]*\\]/,      // Direct recursion in array\n    /&(\\w+)\\s*\\{[^}]*\\*\\1[^}]*\\}/,        // Direct recursion in object\n    /^\\s*\\w+:\\s*&(\\w+)\\s*\\n\\s*\\w+:\\s*\\*\\1/m,  // Multi-line value recursion (data: &ref / value: *ref)\n    \n    // Simplified pattern to detect deeply nested anchors (less ReDoS risk)\n    // Looks for 3+ anchor definitions in close proximity\n    /&\\w+[^&]*&\\w+[^&]*&\\w+/,            // 3+ anchors (simplified, less backtracking)\n    \n    // Detects excessive aliases in close proximity (potential amplification)\n    // Example: [*a, *b, *c, *d, *e, *f, *g, *h, *i, *j]\n    /\\*\\w+(?:[,\\s]+\\*\\w+){9,}/,          // 10+ aliases in sequence (non-capturing group)\n  ];\n\n  private static readonly MALICIOUS_YAML_PATTERNS = [\n    // Language-specific deserialization attacks\n    /!!python\\/object/,\n    /!!python\\/module/,\n    /!!python\\/name/,\n    /!!ruby\\/object/,\n    /!!ruby\\/hash/,\n    /!!ruby\\/struct/,\n    /!!ruby\\/marshal/,\n    /!!java/,\n    /!!javax/,\n    /!!com\\.sun/,\n    /!!perl\\/hash/,\n    /!!perl\\/code/,\n    /!!php\\/object/,\n    \n    // Constructor/function injection\n    /!!exec/,\n    /!!eval/,\n    /!!new/,\n    /!!construct/,\n    /!!apply/,\n    /!!call/,\n    /!!invoke/,\n    \n    // Code execution patterns - more specific to avoid false positives\n    /subprocess\\./,\n    /os\\.system/,\n    /eval\\s*\\(/,\n    /exec\\s*\\(/,\n    /__import__\\s*\\(/,\n    /require\\s*\\(/,\n    /import\\s+(?:os|sys|subprocess|eval|exec)/,\n    /include\\s+[\"'].*\\.(?:php|sh|py|js|rb)[\"']/,\n    \n    // Command execution variants - more specific patterns\n    /popen\\s*\\(/,\n    /spawn\\s*\\(/,\n    /system\\s*\\(/,\n    /backtick\\s*\\(/,\n    /shell_exec\\s*\\(/,\n    /passthru\\s*\\(/,\n    /proc_open\\s*\\(/,\n    \n    // Network operations - require suspicious context\n    /socket\\.connect/,                                      // Detects socket connection attempts\n    /urllib\\.request/,                                      // Python HTTP library usage\n    /requests\\.(?:get|post|put|delete)\\s*\\(/,              // Detects HTTP requests with method calls\n    /fetch\\s*\\(\\s*[\"']https?:\\/\\//,                        // Detects fetch calls to external URLs\n    /new\\s+XMLHttpRequest/,                                 // JavaScript AJAX object creation\n    /\\.(?:get|post|put|delete)\\s*\\(\\s*[\"']https?:\\/\\//,    // Method chaining with HTTP requests\n    \n    // File system operations - require suspicious context\n    /(?:fs\\.|file\\.|)\\s*open\\s*\\(\\s*[\"'](?:\\/etc\\/|\\/bin\\/|\\.\\.\\/)/,     // File open with suspicious paths\n    /file_get_contents\\s*\\(/,                                             // PHP file reading function\n    /file_put_contents\\s*\\(/,                                             // PHP file writing function\n    /fopen\\s*\\(\\s*[\"'](?:\\/etc\\/|\\/bin\\/|\\.\\.\\/)/,                       // File open with dangerous system paths\n    /(?:fs\\.)?\\s*readFile\\s*\\(\\s*[\"'](?:\\/etc\\/|\\/bin\\/|\\.\\.\\/)/,        // Node.js file read with path traversal\n    /(?:fs\\.)?\\s*writeFile\\s*\\(\\s*[\"'](?:\\/(?:bin|etc|tmp)\\/|\\.\\.\\/)/,   // Node.js file write to system dirs\n    \n    // Protocol handlers\n    /file:\\/\\//,\n    /data:\\/\\//,\n    /expect:\\/\\//,\n    /php:\\/\\//,\n    /phar:\\/\\//,\n    /zip:\\/\\//,\n    /ssh2:\\/\\//,\n    /ogg:\\/\\//,\n    \n    // YAML-specific dangerous features\n    /&\\w+\\s*!!/, // Anchor with tag combination\n    /\\*\\w+\\s*!!/, // Alias with tag combination\n    /!!merge/,\n    /!!binary/,\n    /!!timestamp/,\n    \n    // Unicode/encoding bypass attempts - prevent visual spoofing attacks\n    /\\\\[uU]0*(?:22|27|60|3[cC])/,   // Unicode escapes for quotes (\") and brackets (<>)\n    /[\\u202A-\\u202E\\u2066-\\u2069]/,  // Direction override chars (RLO, LRO, isolates)\n    /[\\u200B-\\u200F\\u2028-\\u202F]/,  // Zero-width spaces, line/paragraph separators\n    /[\\uFEFF\\uFFFE\\uFFFF]/,          // BOM, non-characters for payload hiding\n  ];\n\n  /**\n   * Content contexts where code execution patterns are legitimate and should\n   * not trigger security blocks. Skills contain exemplar code; templates contain\n   * code snippets that are rendered, never executed; agent definitions describe\n   * technical workflows that may reference code. Prompt injection, credential,\n   * and path traversal patterns remain active for ALL contexts.\n   * @since Issue #456\n   */\n  private static readonly CODE_EXEMPT_CONTEXTS = new Set<ContentValidatorOptions['contentContext']>([\n    'skill',    // Exemplar code patterns the LLM should follow\n    'template', // Code snippets rendered into output, never executed\n    'agent',    // Technical workflow definitions — without this, agents would need\n                // to pull in a skill or template just to reference code, adding\n                // coupling without security value. Agent definitions are authored\n                // content read as LLM context, same as skills and templates.\n  ]);\n\n  /**\n   * Pattern descriptions that are exempt for CODE_EXEMPT_CONTEXTS.\n   * These patterns match legitimate code documentation, not threats.\n   * @since Issue #456\n   */\n  private static readonly CODE_EXECUTION_PATTERNS = new Set([\n    'Code evaluation',\n    'Code execution',\n    'System command execution',\n    'Subprocess execution',\n  ]);\n\n  /**\n   * HTML/XSS pattern descriptions exempt for template context.\n   * Templates use <template>, <style>, <script> as section delimiters.\n   * @since Issue #803\n   */\n  private static readonly HTML_SECTION_PATTERNS = new Set([\n    'HTML script injection',\n    'HTML object injection',\n    'HTML embed injection',\n  ]);\n\n  /**\n   * Handles Unicode validation and threat detection\n   * REFACTOR: Extracted from validateAndSanitize() to reduce cognitive complexity\n   * Returns normalized content and Unicode severity without aborting early\n   */\n  private static handleUnicodeValidation(\n    content: string,\n    detectedPatterns: string[]\n  ): {\n    sanitized: string;\n    highestSeverity: SecuritySeverity;\n  } {\n    const unicodeResult = UnicodeValidator.normalize(content);\n    const sanitized = unicodeResult.normalizedContent;\n    let highestSeverity: SecuritySeverity = 'low';\n\n    if (!unicodeResult.isValid && unicodeResult.detectedIssues) {\n      detectedPatterns.push(...unicodeResult.detectedIssues.map(issue => `Unicode: ${issue}`));\n      if (unicodeResult.severity) {\n        highestSeverity = unicodeResult.severity;\n      }\n\n      // Log high/critical Unicode attacks\n      if (unicodeResult.severity === 'critical' || unicodeResult.severity === 'high') {\n        SecurityMonitor.logSecurityEvent({\n          type: 'CONTENT_INJECTION_ATTEMPT',\n          severity: unicodeResult.severity.toUpperCase() as 'HIGH' | 'CRITICAL',\n          source: 'content_validation',\n          details: `Unicode attack detected: ${unicodeResult.detectedIssues.join(', ')}`,\n        });\n\n        ContentValidator.getTelemetry()?.recordBlockedAttack(\n          'UNICODE_ATTACK',\n          unicodeResult.detectedIssues.join(', '),\n          unicodeResult.severity.toUpperCase() as 'HIGH' | 'CRITICAL',\n          'unicode_validation',\n          { issues: unicodeResult.detectedIssues }\n        );\n      }\n    }\n\n    return { sanitized, highestSeverity };\n  }\n\n  /**\n   * Checks content for injection patterns and logs/sanitizes threats\n   * REFACTOR: Extracted from validateAndSanitize() to reduce cognitive complexity\n   *\n   * @param originalContent - Original content to check patterns against\n   * @param normalizedContent - Normalized content to apply replacements to\n   * @param detectedPatterns - Array to accumulate detected pattern descriptions\n   * @param currentSeverity - Current highest severity level\n   * @param maxLength - Maximum allowed content length for regex validation\n   */\n  private static checkInjectionPatterns(\n    originalContent: string,\n    normalizedContent: string,\n    detectedPatterns: string[],\n    currentSeverity: SecuritySeverity,\n    maxLength: number,\n    contentContext?: ContentValidatorOptions['contentContext']\n  ): {\n    sanitized: string;\n    highestSeverity: SecuritySeverity;\n  } {\n    let sanitized = normalizedContent;\n    let highestSeverity = currentSeverity;\n\n    for (const { pattern, severity, description } of this.INJECTION_PATTERNS) {\n      // Fix #456: Skip code execution patterns for element types that legitimately contain code\n      if (contentContext && this.CODE_EXEMPT_CONTEXTS.has(contentContext) && this.CODE_EXECUTION_PATTERNS.has(description)) {\n        continue;\n      }\n      // Fix #803: Skip HTML section tag patterns for templates (use <script>/<style> as section delimiters)\n      if (contentContext === 'template' && this.HTML_SECTION_PATTERNS.has(description)) {\n        continue;\n      }\n      // Check pattern on original content (before normalization) to catch encoded attacks\n      if (RegexValidator.validate(originalContent, pattern, {\n        maxLength,\n        rejectDangerousPatterns: false,\n        logEvents: false\n      })) {\n        detectedPatterns.push(description);\n\n        // Update highest severity\n        if (severity === 'critical' || (severity === 'high' && highestSeverity !== 'critical')) {\n          highestSeverity = severity;\n        }\n\n        // Log security event\n        SecurityMonitor.logSecurityEvent({\n          type: 'CONTENT_INJECTION_ATTEMPT',\n          severity: severity.toUpperCase() as 'HIGH' | 'CRITICAL',\n          source: 'content_validation',\n          details: `Detected pattern: ${description}`,\n        });\n\n        // Record in telemetry\n        ContentValidator.getTelemetry()?.recordBlockedAttack(\n          'CONTENT_INJECTION',\n          description,\n          severity.toUpperCase() as 'HIGH' | 'CRITICAL',\n          'content_validation',\n          { pattern: pattern.source }\n        );\n\n        // Apply replacement to normalized content\n        sanitized = sanitized.replace(pattern, '[CONTENT_BLOCKED]');\n      }\n    }\n\n    return { sanitized, highestSeverity };\n  }\n\n  /**\n   * Validates and sanitizes persona content for security threats\n   * FIX #1269: Added options to support large memory content\n   * REFACTOR: Reduced cognitive complexity by extracting helper methods\n   *\n   * SECURITY FIX (DMCP-SEC-004): Length checks now performed on NORMALIZED content\n   * to prevent bypass attacks using Unicode combining characters or zero-width chars.\n   * A pre-check with generous multiplier prevents DoS from huge payloads.\n   */\n  static validateAndSanitize(content: string, options: ContentValidatorOptions = {}): ContentValidationResult {\n    // Determine max length for validation\n    const maxLength = options.maxLength || SECURITY_LIMITS.MAX_CONTENT_LENGTH;\n\n    // SECURITY FIX (DMCP-SEC-004): Two-phase length validation\n    // Phase 1: DoS prevention pre-check on raw content (generous 2x multiplier)\n    // This prevents huge payloads from hitting the normalization code path\n    // while still allowing legitimate content with some Unicode overhead\n    const DOS_PREVENTION_MULTIPLIER = 2;\n    if (!options.skipSizeCheck) {\n      if (content.length > maxLength * DOS_PREVENTION_MULTIPLIER) {\n        throw new SecurityError(\n          `Content exceeds maximum length of ${maxLength} characters (${content.length} provided)`\n        );\n      }\n    }\n\n    const detectedPatterns: string[] = [];\n\n    // Handle Unicode validation (normalizes content but doesn't abort)\n    const unicodeCheck = this.handleUnicodeValidation(content, detectedPatterns);\n\n    // SECURITY FIX (DMCP-SEC-004): Phase 2 - Check length on NORMALIZED content\n    // This prevents bypass attacks using combining characters or zero-width chars\n    // that would inflate raw length but collapse after normalization\n    if (!options.skipSizeCheck) {\n      if (unicodeCheck.sanitized.length > maxLength) {\n        throw new SecurityError(\n          `Content exceeds maximum length of ${maxLength} characters after normalization (${unicodeCheck.sanitized.length} provided)`\n        );\n      }\n    }\n\n    // Check for injection patterns on ORIGINAL content (to catch encoded attacks)\n    // but apply replacements to NORMALIZED content (to preserve normalization)\n    const injectionCheck = this.checkInjectionPatterns(\n      content,\n      unicodeCheck.sanitized,\n      detectedPatterns,\n      unicodeCheck.highestSeverity,\n      maxLength,\n      options.contentContext\n    );\n\n    // Use highest severity from either Unicode or injection checks\n    const finalSeverity = injectionCheck.highestSeverity;\n\n    // Abort if high/critical threats detected\n    if (finalSeverity === 'high' || finalSeverity === 'critical') {\n      return {\n        isValid: false,\n        sanitizedContent: injectionCheck.sanitized,\n        detectedPatterns,\n        severity: finalSeverity\n      };\n    }\n\n    return {\n      isValid: detectedPatterns.length === 0,\n      sanitizedContent: injectionCheck.sanitized,\n      detectedPatterns,\n      severity: finalSeverity\n    };\n  }\n\n  /**\n   * Validates YAML frontmatter for malicious content\n   * SECURITY FIX #364: Added YAML bomb detection to prevent denial of service\n   */\n  static validateYamlContent(yamlContent: string): boolean {\n    // Length validation before pattern matching\n    if (yamlContent.length > SECURITY_LIMITS.MAX_YAML_LENGTH) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'YAML_INJECTION_ATTEMPT',\n        severity: 'HIGH',\n        source: 'yaml_validation',\n        details: `YAML content exceeds maximum length: ${yamlContent.length} > ${SECURITY_LIMITS.MAX_YAML_LENGTH}`\n      });\n      return false;\n    }\n\n    // SECURITY FIX #364: Check for YAML bombs before other validation\n    // SECURITY FIX (PR #552 review): Use RegexValidator for ReDoS protection\n    for (const pattern of this.YAML_BOMB_PATTERNS) {\n      // Use RegexValidator to safely check patterns with timeout protection\n      // This prevents ReDoS attacks from maliciously crafted YAML\n      const isMatch = RegexValidator.validate(yamlContent, pattern, {\n        maxLength: SECURITY_LIMITS.MAX_YAML_LENGTH,\n        rejectDangerousPatterns: false, // Our patterns are trusted\n        logEvents: false // We handle logging ourselves\n      });\n      \n      if (isMatch) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'YAML_INJECTION_ATTEMPT',\n          severity: 'CRITICAL',\n          source: 'yaml_bomb_detection',\n          details: `YAML bomb pattern detected: ${pattern.source}`,\n          metadata: {\n            patternType: 'YAML_BOMB',\n            contentLength: yamlContent.length\n          }\n        });\n\n        // Record in telemetry\n        ContentValidator.getTelemetry()?.recordBlockedAttack(\n          'YAML_BOMB',\n          `YAML bomb pattern: ${pattern.source}`,\n          'CRITICAL',\n          'yaml_validation',\n          { patternType: 'YAML_BOMB', contentLength: yamlContent.length }\n        );\n\n        return false;\n      }\n    }\n    \n    // SECURITY FIX #364: Count anchor/alias ratio for amplification detection\n    // SECURITY FIX #1298: Use configurable threshold for easier tuning\n    const anchorMatches = yamlContent.match(/&\\w+/g) || [];\n    // Fix #906: Use negative lookbehind to exclude markdown bold (**word**) from\n    // matching as YAML aliases. Without this, markdown bold inside YAML strings\n    // triggers false-positive amplification detection.\n    const aliasMatches = yamlContent.match(/(?<!\\*)\\*\\w+/g) || [];\n    const amplificationRatio = anchorMatches.length > 0 ? aliasMatches.length / anchorMatches.length : 0;\n\n    if (amplificationRatio > SECURITY_LIMITS.YAML_BOMB_AMPLIFICATION_THRESHOLD) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'YAML_INJECTION_ATTEMPT',\n        severity: 'HIGH',\n        source: 'yaml_amplification_detection',\n        details: `Excessive alias amplification detected: ${aliasMatches.length} aliases for ${anchorMatches.length} anchors (ratio: ${amplificationRatio.toFixed(2)})`,\n        metadata: {\n          anchors: anchorMatches.length,\n          aliases: aliasMatches.length,\n          ratio: amplificationRatio\n        }\n      });\n      return false;\n    }\n    \n    // SECURITY FIX #364: Detect circular reference chains\n    // SECURITY FIX (PR #552 review): Optimized from O(n²) to O(n) using Set-based lookups\n    const anchorRefs = new Map<string, Set<string>>();\n    const lines = yamlContent.split('\\n');\n    \n    // First pass: Build reference map efficiently\n    for (let i = 0; i < lines.length; i++) {\n      const anchorMatch = lines[i].match(/&(\\w+)/);\n      if (anchorMatch) {\n        const anchorName = anchorMatch[1];\n        // Get references in next 5 lines\n        const contextEnd = Math.min(i + 5, lines.length);\n        const references = new Set<string>();\n        \n        for (let j = i; j < contextEnd; j++) {\n          const aliasMatches = lines[j].match(/\\*(\\w+)/g);\n          if (aliasMatches) {\n            aliasMatches.forEach(alias => {\n              references.add(alias.substring(1)); // Remove * prefix\n            });\n          }\n        }\n        \n        anchorRefs.set(anchorName, references);\n      }\n    }\n    \n    // Second pass: Check for circular references (O(n) with Set lookups)\n    for (const [anchor1, refs1] of anchorRefs) {\n      for (const refAnchor of refs1) {\n        const refs2 = anchorRefs.get(refAnchor);\n        // Check if the referenced anchor references back to the original\n        if (refs2 && refs2.has(anchor1)) {\n          SecurityMonitor.logSecurityEvent({\n            type: 'YAML_INJECTION_ATTEMPT',\n            severity: 'CRITICAL',\n            source: 'yaml_bomb_detection',\n            details: `Circular reference chain detected between anchors: &${anchor1} and &${refAnchor}`,\n            metadata: {\n              patternType: 'CIRCULAR_REFERENCE',\n              anchors: [anchor1, refAnchor]\n            }\n          });\n          return false;\n        }\n      }\n    }\n    \n    // Unicode normalization preprocessing for YAML content\n    const unicodeResult = UnicodeValidator.normalize(yamlContent);\n    const normalizedYaml = unicodeResult.normalizedContent;\n    \n    if (!unicodeResult.isValid && unicodeResult.detectedIssues) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'YAML_UNICODE_ATTACK',\n        severity: (unicodeResult.severity?.toUpperCase() || 'MEDIUM') as 'LOW' | 'MEDIUM' | 'HIGH' | 'CRITICAL',\n        source: 'yaml_validation',\n        details: `Unicode attack detected in YAML: ${unicodeResult.detectedIssues.join(', ')}`\n      });\n      return false;\n    }\n\n    for (const pattern of this.MALICIOUS_YAML_PATTERNS) {\n      // These are trusted internal patterns, so we disable ReDoS rejection\n      if (RegexValidator.validate(normalizedYaml, pattern, {\n        maxLength: SECURITY_LIMITS.MAX_CONTENT_LENGTH,\n        rejectDangerousPatterns: false,\n        logEvents: false  // Don't log our own security patterns as dangerous\n      })) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'YAML_INJECTION_ATTEMPT',\n          severity: 'CRITICAL',\n          source: 'yaml_validation',\n          details: `Malicious YAML pattern detected: ${pattern}`,\n        });\n        // Early exit on first match for performance\n        return false;\n      }\n    }\n    return true;\n  }\n\n  /**\n   * Validates persona metadata fields\n   */\n  static validateMetadata(metadata: any): ContentValidationResult {\n    const detectedPatterns: string[] = [];\n\n    // Check all string fields in metadata\n    const checkField = (fieldName: string, value: any) => {\n      if (typeof value === 'string') {\n        // Check field length first\n        if (value.length > SECURITY_LIMITS.MAX_METADATA_FIELD_LENGTH) {\n          detectedPatterns.push(`${fieldName}: Field exceeds maximum length of ${SECURITY_LIMITS.MAX_METADATA_FIELD_LENGTH} characters`);\n          return;\n        }\n        \n        const result = this.validateAndSanitize(value);\n        if (!result.isValid || result.detectedPatterns?.length) {\n          detectedPatterns.push(`${fieldName}: ${result.detectedPatterns?.join(', ')}`);\n        }\n      }\n    };\n\n    // Validate standard persona fields\n    checkField('name', metadata.name);\n    checkField('description', metadata.description);\n    checkField('category', metadata.category);\n    checkField('author', metadata.author);\n    \n    // Check any custom fields\n    for (const [key, value] of Object.entries(metadata)) {\n      if (!['name', 'description', 'category', 'author'].includes(key)) {\n        checkField(key, value);\n      }\n    }\n\n    return {\n      isValid: detectedPatterns.length === 0,\n      detectedPatterns,\n      severity: detectedPatterns.length > 0 ? 'high' : 'low'\n    };\n  }\n\n  /**\n   * Sanitizes a complete persona file (frontmatter + content)\n   */\n  static sanitizePersonaContent(content: string): string {\n    // Extract frontmatter\n    const frontmatterMatch = content.match(/^---\\n([\\s\\S]*?)\\n---/);\n    \n    if (!frontmatterMatch) {\n      // No frontmatter, just validate content\n      const result = this.validateAndSanitize(content);\n      if (!result.isValid && result.severity === 'critical') {\n        // FIX: Include specific patterns that triggered the rejection for debugging\n        const patterns = result.detectedPatterns?.join(', ') || 'unknown patterns';\n        throw new SecurityError(`Critical security threat detected in persona content: ${patterns}`);\n      }\n      return result.sanitizedContent || content;\n    }\n\n    const yamlContent = frontmatterMatch[1];\n    const markdownContent = content.substring(frontmatterMatch[0].length);\n\n    // Validate YAML\n    if (!this.validateYamlContent(yamlContent)) {\n      throw new SecurityError('Malicious YAML detected in persona frontmatter');\n    }\n\n    // Validate markdown content\n    const contentResult = this.validateAndSanitize(markdownContent);\n    if (!contentResult.isValid && contentResult.severity === 'critical') {\n      // FIX: Include specific patterns that triggered the rejection for debugging\n      const patterns = contentResult.detectedPatterns?.join(', ') || 'unknown patterns';\n      throw new SecurityError(`Critical security threat detected in persona content: ${patterns}`);\n    }\n\n    // Return sanitized content\n    return `---\\n${yamlContent}\\n---${contentResult.sanitizedContent || markdownContent}`;\n  }\n}"]}
601
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"contentValidator.js","sourceRoot":"","sources":["../../src/security/contentValidator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Content Validator for DollhouseMCP\n * \n * Protects against prompt injection attacks in collection personas\n * by detecting and sanitizing malicious content patterns.\n * \n * Security: SEC-001 - Critical vulnerability protection\n */\n\nimport { SecurityError } from './errors.js';\nimport { SecurityMonitor } from './securityMonitor.js';\nimport { RegexValidator } from './regexValidator.js';\nimport { SECURITY_LIMITS } from './constants.js';\nimport { UnicodeValidator } from './validators/unicodeValidator.js';\nimport type { SecurityTelemetry } from './telemetry/SecurityTelemetry.js';\n\n// FIX: SonarCloud typescript:S4323 - Extract union type to type alias for maintainability\nexport type SecuritySeverity = 'low' | 'medium' | 'high' | 'critical';\n\nexport interface ContentValidationResult {\n  isValid: boolean;\n  sanitizedContent?: string;\n  detectedPatterns?: string[];\n  severity?: SecuritySeverity;\n}\n\nexport interface ContentValidatorOptions {\n  /**\n   * Skip size limit checks - useful for memory content that can be large\n   * @default false\n   */\n  skipSizeCheck?: boolean;\n  /**\n   * Custom max length override\n   * @default SECURITY_LIMITS.MAX_CONTENT_LENGTH\n   */\n  maxLength?: number;\n  /**\n   * Element type context for context-aware pattern matching.\n   * Skills may legitimately contain code patterns (eval, exec, require)\n   * that would be blocked in other contexts.\n   * @since Issue #456\n   */\n  contentContext?: 'persona' | 'skill' | 'template' | 'agent' | 'memory';\n}\n\nexport class ContentValidator {\n  private static telemetryResolver?: () => SecurityTelemetry | undefined;\n\n  public static configureTelemetryResolver(resolver: () => SecurityTelemetry | undefined): void {\n    this.telemetryResolver = resolver;\n  }\n\n  private static getTelemetry(): SecurityTelemetry | undefined {\n    try {\n      return this.telemetryResolver ? this.telemetryResolver() : undefined;\n    } catch {\n      return undefined;\n    }\n  }\n  /**\n   * Pattern-based detection system for prompt injection attacks.\n   * \n   * This approach was chosen over AI-based detection because:\n   * 1. Pattern matching cannot be socially engineered or confused\n   * 2. Deterministic results ensure consistent security\n   * 3. No additional API calls or latency\n   * 4. Can't be bypassed by clever prompt engineering\n   * \n   * The patterns below represent known attack vectors from security research\n   * and real-world exploit attempts against AI systems.\n   */\n  // Prompt injection patterns that could compromise AI assistants\n  private static readonly INJECTION_PATTERNS: Array<{ pattern: RegExp; severity: 'high' | 'critical'; description: string }> = [\n    // System prompt override attempts\n    { pattern: /\\[SYSTEM:\\s*.*?\\]/gi, severity: 'critical', description: 'System prompt override' },\n    { pattern: /\\[ADMIN:\\s*.*?\\]/gi, severity: 'critical', description: 'Admin prompt override' },\n    { pattern: /\\[ASSISTANT:\\s*.*?\\]/gi, severity: 'critical', description: 'Assistant prompt override' },\n    { pattern: /\\[USER:\\s*.*?\\]/gi, severity: 'high', description: 'User prompt override' },\n    \n    // Instruction manipulation\n    { pattern: /ignore\\s+(all\\s+)?previous\\s+instructions/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /ignore\\s+(all\\s+)?prior\\s+instructions/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /disregard\\s+(all\\s+)?previous\\s+instructions/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /disregard\\s+everything\\s+above/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /forget\\s+(all\\s+)?previous\\s+instructions/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /forget\\s+your\\s+training/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /override\\s+your\\s+programming/gi, severity: 'critical', description: 'Instruction override' },\n    { pattern: /you\\s+are\\s+now\\s+(admin|root|system|sudo)/gi, severity: 'critical', description: 'Role elevation attempt' },\n    { pattern: /act\\s+as\\s+(admin|root|system|sudo)/gi, severity: 'critical', description: 'Role elevation attempt' },\n    \n    // Data exfiltration attempts\n    { pattern: /export\\s+all\\s+(files|data|personas|tokens|credentials|api\\s+keys)/gi, severity: 'critical', description: 'Data exfiltration' },\n    { pattern: /send\\s+all\\s+(files|data|personas|tokens|credentials|api\\s+keys)\\s+to/gi, severity: 'critical', description: 'Data exfiltration' },\n    { pattern: /list\\s+all\\s+(files|tokens|credentials|secrets|api\\s+keys)/gi, severity: 'high', description: 'Information disclosure' },\n    { pattern: /show\\s+me\\s+all\\s+(tokens|credentials|secrets|api\\s+keys)/gi, severity: 'high', description: 'Credential disclosure' },\n    \n    // Command execution patterns\n    { pattern: /curl\\s+[^\\s]+\\.(com|net|org|io|dev)/gi, severity: 'critical', description: 'External command execution' },\n    { pattern: /wget\\s+[^\\s]+\\.(com|net|org|io|dev)/gi, severity: 'critical', description: 'External command execution' },\n    { pattern: /\\$\\([^)]+\\)/g, severity: 'critical', description: 'Command substitution' },\n    // SECURITY: Backtick command detection with ReDoS mitigation\n    // FIX (PR #1313): Fixed ReDoS vulnerabilities by replacing .* with [^`]*\n    // FIX (PR #1313 - SonarCloud): Added explicit bounds {0,200} to prevent backtracking\n    // Multiple unbounded quantifiers in same pattern can still cause backtracking even with [^`]*\n    // Bounded quantifiers prevent exponential time complexity while matching realistic commands\n    { pattern: /`[^`]{0,200}(?:rm\\s+-rf?\\s+[/~]|sudo\\s+rm|chmod\\s+777|chown\\s+root)[^`]{0,200}`/gi, severity: 'critical', description: 'Dangerous shell command in backticks' },\n    { pattern: /`[^`]{0,200}(?:cat|ls)\\s+\\/etc\\/[^`]{0,200}`/gi, severity: 'critical', description: 'Sensitive file access in backticks' },\n    { pattern: /`[^`]{0,200}(?:bash|sh)\\s+-c\\s+['\"][^`]{0,200}`/gi, severity: 'critical', description: 'Shell execution in backticks' },\n    { pattern: /`[^`]{0,200}(?:passwd|shadow|nc\\s+-l|netcat\\s+-l|ssh\\s+root@)[^`]{0,200}`/gi, severity: 'critical', description: 'Dangerous command in backticks' },\n    { pattern: /`[^`]{0,200}(?:curl|wget)\\s+[^`]{0,200}\\|\\s*(?:sh|bash)[^`]{0,200}`/gi, severity: 'critical', description: 'Pipe to shell in backticks' },\n    { pattern: /`[^`]{0,200}(?:\\/etc\\/passwd|\\/etc\\/shadow|\\.ssh\\/id_|sudo\\s+su)[^`]{0,200}`/gi, severity: 'critical', description: 'Sensitive file or privilege escalation in backticks' },\n    { pattern: /`[^`]{0,200}(?:python|perl|ruby|php|node)\\s+(?:-e|-c)\\s+[^`]{0,200}(?:exec|eval|system|subprocess)[^`]{0,200}`/gi, severity: 'critical', description: 'Script interpreter with dangerous function in backticks' },\n    { pattern: /eval\\s*\\(/gi, severity: 'critical', description: 'Code evaluation' },\n    { pattern: /exec\\s*\\(/gi, severity: 'critical', description: 'Code execution' },\n    { pattern: /os\\.system\\s*\\(/gi, severity: 'critical', description: 'System command execution' },\n    { pattern: /subprocess\\.(call|run|Popen)/gi, severity: 'critical', description: 'Subprocess execution' },\n    \n    // Token/credential patterns\n    { pattern: /GITHUB_TOKEN/gi, severity: 'high', description: 'Token reference' },\n    { pattern: /ghp_[a-zA-Z0-9]{36}/g, severity: 'critical', description: 'GitHub token exposure' },\n    { pattern: /gho_[a-zA-Z0-9]{36}/g, severity: 'critical', description: 'GitHub OAuth token exposure' },\n    \n    // Path traversal in content\n    { pattern: /\\.\\.\\/\\.\\.\\/\\.\\.\\//g, severity: 'high', description: 'Path traversal attempt' },\n    { pattern: /\\/etc\\/passwd/gi, severity: 'high', description: 'Sensitive file access' },\n    { pattern: /\\/\\.ssh\\//gi, severity: 'high', description: 'SSH key access attempt' },\n\n    // HTML/XSS patterns — defense-in-depth for community-sourced content\n    // DOMPurify on the client is the primary defense; these catch threats at ingest\n    { pattern: /<script[\\s>]/gi, severity: 'critical', description: 'HTML script injection' },\n    { pattern: /<\\/script>/gi, severity: 'critical', description: 'HTML script injection' },\n    { pattern: /<iframe[\\s>]/gi, severity: 'critical', description: 'HTML iframe injection' },\n    { pattern: /<object[\\s>]/gi, severity: 'high', description: 'HTML object injection' },\n    { pattern: /<embed[\\s>]/gi, severity: 'high', description: 'HTML embed injection' },\n    { pattern: /\\bon\\w+=\\s*[\"']/gi, severity: 'critical', description: 'HTML event handler injection' },\n    { pattern: /javascript\\s*:/gi, severity: 'critical', description: 'JavaScript protocol injection' },\n    // Entity-encoded variants: &#106;avascript, &#x6a;avascript, &#106;&#97;vascript, etc.\n    { pattern: /&#x?[0-9a-f]+;?\\s*a\\s*v\\s*a\\s*s\\s*c\\s*r\\s*i\\s*p\\s*t/gi, severity: 'critical', description: 'Encoded JavaScript protocol injection' },\n    // Fully/partially entity-encoded: detects &#...script pattern (covers multi-entity encoding)\n    { pattern: /(?:&#x?[0-9a-f]+;?\\s*){2,}s\\s*c\\s*r\\s*i\\s*p\\s*t/gi, severity: 'critical', description: 'Encoded JavaScript protocol injection' },\n  ];\n\n  // Malicious YAML patterns\n  // SECURITY FIX #364: YAML bomb detection patterns\n  // SECURITY FIX (PR #552 review): Simplified patterns to reduce ReDoS risk\n  private static readonly YAML_BOMB_PATTERNS = [\n    // Detects recursive anchor references that could cause exponential expansion\n    // Example: &a [*a] or &bomb [\"test\", *bomb]\n    /&(\\w+)\\s*\\[[^\\]]*\\*\\1[^\\]]*\\]/,      // Direct recursion in array\n    /&(\\w+)\\s*\\{[^}]*\\*\\1[^}]*\\}/,        // Direct recursion in object\n    /^\\s*\\w+:\\s*&(\\w+)\\s*\\n\\s*\\w+:\\s*\\*\\1/m,  // Multi-line value recursion (data: &ref / value: *ref)\n    \n    // Simplified pattern to detect deeply nested anchors (less ReDoS risk)\n    // Looks for 3+ anchor definitions in close proximity\n    /&\\w+[^&]*&\\w+[^&]*&\\w+/,            // 3+ anchors (simplified, less backtracking)\n    \n    // Detects excessive aliases in close proximity (potential amplification)\n    // Example: [*a, *b, *c, *d, *e, *f, *g, *h, *i, *j]\n    /\\*\\w+(?:[,\\s]+\\*\\w+){9,}/,          // 10+ aliases in sequence (non-capturing group)\n  ];\n\n  private static readonly MALICIOUS_YAML_PATTERNS = [\n    // Language-specific deserialization attacks\n    /!!python\\/object/,\n    /!!python\\/module/,\n    /!!python\\/name/,\n    /!!ruby\\/object/,\n    /!!ruby\\/hash/,\n    /!!ruby\\/struct/,\n    /!!ruby\\/marshal/,\n    /!!java/,\n    /!!javax/,\n    /!!com\\.sun/,\n    /!!perl\\/hash/,\n    /!!perl\\/code/,\n    /!!php\\/object/,\n    \n    // Constructor/function injection\n    /!!exec/,\n    /!!eval/,\n    /!!new/,\n    /!!construct/,\n    /!!apply/,\n    /!!call/,\n    /!!invoke/,\n    \n    // Code execution patterns - more specific to avoid false positives\n    /subprocess\\./,\n    /os\\.system/,\n    /eval\\s*\\(/,\n    /exec\\s*\\(/,\n    /__import__\\s*\\(/,\n    /require\\s*\\(/,\n    /import\\s+(?:os|sys|subprocess|eval|exec)/,\n    /include\\s+[\"'].*\\.(?:php|sh|py|js|rb)[\"']/,\n    \n    // Command execution variants - more specific patterns\n    /popen\\s*\\(/,\n    /spawn\\s*\\(/,\n    /system\\s*\\(/,\n    /backtick\\s*\\(/,\n    /shell_exec\\s*\\(/,\n    /passthru\\s*\\(/,\n    /proc_open\\s*\\(/,\n    \n    // Network operations - require suspicious context\n    /socket\\.connect/,                                      // Detects socket connection attempts\n    /urllib\\.request/,                                      // Python HTTP library usage\n    /requests\\.(?:get|post|put|delete)\\s*\\(/,              // Detects HTTP requests with method calls\n    /fetch\\s*\\(\\s*[\"']https?:\\/\\//,                        // Detects fetch calls to external URLs\n    /new\\s+XMLHttpRequest/,                                 // JavaScript AJAX object creation\n    /\\.(?:get|post|put|delete)\\s*\\(\\s*[\"']https?:\\/\\//,    // Method chaining with HTTP requests\n    \n    // File system operations - require suspicious context\n    /(?:fs\\.|file\\.|)\\s*open\\s*\\(\\s*[\"'](?:\\/etc\\/|\\/bin\\/|\\.\\.\\/)/,     // File open with suspicious paths\n    /file_get_contents\\s*\\(/,                                             // PHP file reading function\n    /file_put_contents\\s*\\(/,                                             // PHP file writing function\n    /fopen\\s*\\(\\s*[\"'](?:\\/etc\\/|\\/bin\\/|\\.\\.\\/)/,                       // File open with dangerous system paths\n    /(?:fs\\.)?\\s*readFile\\s*\\(\\s*[\"'](?:\\/etc\\/|\\/bin\\/|\\.\\.\\/)/,        // Node.js file read with path traversal\n    /(?:fs\\.)?\\s*writeFile\\s*\\(\\s*[\"'](?:\\/(?:bin|etc|tmp)\\/|\\.\\.\\/)/,   // Node.js file write to system dirs\n    \n    // Protocol handlers\n    /file:\\/\\//,\n    /data:\\/\\//,\n    /expect:\\/\\//,\n    /php:\\/\\//,\n    /phar:\\/\\//,\n    /zip:\\/\\//,\n    /ssh2:\\/\\//,\n    /ogg:\\/\\//,\n    \n    // YAML-specific dangerous features\n    /&\\w+\\s*!!/, // Anchor with tag combination\n    /\\*\\w+\\s*!!/, // Alias with tag combination\n    /!!merge/,\n    /!!binary/,\n    /!!timestamp/,\n    \n    // Unicode/encoding bypass attempts - prevent visual spoofing attacks\n    /\\\\[uU]0*(?:22|27|60|3[cC])/,   // Unicode escapes for quotes (\") and brackets (<>)\n    /[\\u202A-\\u202E\\u2066-\\u2069]/,  // Direction override chars (RLO, LRO, isolates)\n    /[\\u200B-\\u200F\\u2028-\\u202F]/,  // Zero-width spaces, line/paragraph separators\n    /[\\uFEFF\\uFFFE\\uFFFF]/,          // BOM, non-characters for payload hiding\n  ];\n\n  /**\n   * Content contexts where code execution patterns are legitimate and should\n   * not trigger security blocks. Skills contain exemplar code; templates contain\n   * code snippets that are rendered, never executed; agent definitions describe\n   * technical workflows that may reference code. Prompt injection, actual token\n   * exposure (ghp_/gho_), data exfiltration, and HTML/XSS patterns remain\n   * active for ALL contexts.\n   * @since Issue #456\n   */\n  private static readonly CODE_EXEMPT_CONTEXTS = new Set<ContentValidatorOptions['contentContext']>([\n    'skill',    // Exemplar code patterns the LLM should follow\n    'template', // Code snippets rendered into output, never executed\n    'agent',    // Technical workflow definitions — without this, agents would need\n                // to pull in a skill or template just to reference code, adding\n                // coupling without security value. Agent definitions are authored\n                // content read as LLM context, same as skills and templates.\n  ]);\n\n  /**\n   * Pattern descriptions that are exempt for CODE_EXEMPT_CONTEXTS.\n   * These patterns match legitimate code documentation, not threats.\n   * @since Issue #456\n   */\n  private static readonly CODE_EXECUTION_PATTERNS = new Set([\n    'Code evaluation',\n    'Code execution',\n    'System command execution',\n    'Subprocess execution',\n  ]);\n\n  /**\n   * Security documentation patterns exempt for CODE_EXEMPT_CONTEXTS.\n   * Skills/agents that teach penetration testing, threat modeling, etc.\n   * legitimately reference shell commands, file paths, and credential names.\n   *\n   * DISTINCTION FROM ACTIVE THREAT PATTERNS:\n   * These patterns describe attacks (educational) — they appear in element\n   * definitions that an author wrote, not in runtime user input. Patterns\n   * that remain active even in exempt contexts are actual threats:\n   * - Prompt injection (system/admin override, instruction manipulation)\n   * - Real token formats (ghp_*, gho_* — not just the word \"GITHUB_TOKEN\")\n   * - Data exfiltration commands (export/send all credentials)\n   * - HTML/XSS injection (renders in the web console)\n   *\n   * @since Issue #1725\n   */\n  private static readonly SECURITY_DOC_PATTERNS = new Set([\n    'Command substitution',\n    'External command execution',\n    'Sensitive file access',\n    'Path traversal attempt',\n    'SSH key access attempt',\n    'Token reference',\n    'Dangerous shell command in backticks',\n    'Sensitive file access in backticks',\n    'Shell execution in backticks',\n    'Dangerous command in backticks',\n    'Pipe to shell in backticks',\n    'Sensitive file or privilege escalation in backticks',\n    'Script interpreter with dangerous function in backticks',\n  ]);\n\n  /**\n   * HTML/XSS pattern descriptions exempt for template context.\n   * Templates use <template>, <style>, <script> as section delimiters.\n   * @since Issue #803\n   */\n  private static readonly HTML_SECTION_PATTERNS = new Set([\n    'HTML script injection',\n    'HTML object injection',\n    'HTML embed injection',\n  ]);\n\n  /**\n   * Handles Unicode validation and threat detection\n   * REFACTOR: Extracted from validateAndSanitize() to reduce cognitive complexity\n   * Returns normalized content and Unicode severity without aborting early\n   */\n  private static handleUnicodeValidation(\n    content: string,\n    detectedPatterns: string[]\n  ): {\n    sanitized: string;\n    highestSeverity: SecuritySeverity;\n  } {\n    const unicodeResult = UnicodeValidator.normalize(content);\n    const sanitized = unicodeResult.normalizedContent;\n    let highestSeverity: SecuritySeverity = 'low';\n\n    if (!unicodeResult.isValid && unicodeResult.detectedIssues) {\n      detectedPatterns.push(...unicodeResult.detectedIssues.map(issue => `Unicode: ${issue}`));\n      if (unicodeResult.severity) {\n        highestSeverity = unicodeResult.severity;\n      }\n\n      // Log high/critical Unicode attacks\n      if (unicodeResult.severity === 'critical' || unicodeResult.severity === 'high') {\n        SecurityMonitor.logSecurityEvent({\n          type: 'CONTENT_INJECTION_ATTEMPT',\n          severity: unicodeResult.severity.toUpperCase() as 'HIGH' | 'CRITICAL',\n          source: 'content_validation',\n          details: `Unicode attack detected: ${unicodeResult.detectedIssues.join(', ')}`,\n        });\n\n        ContentValidator.getTelemetry()?.recordBlockedAttack(\n          'UNICODE_ATTACK',\n          unicodeResult.detectedIssues.join(', '),\n          unicodeResult.severity.toUpperCase() as 'HIGH' | 'CRITICAL',\n          'unicode_validation',\n          { issues: unicodeResult.detectedIssues }\n        );\n      }\n    }\n\n    return { sanitized, highestSeverity };\n  }\n\n  /**\n   * Checks content for injection patterns and logs/sanitizes threats\n   * REFACTOR: Extracted from validateAndSanitize() to reduce cognitive complexity\n   *\n   * @param originalContent - Original content to check patterns against\n   * @param normalizedContent - Normalized content to apply replacements to\n   * @param detectedPatterns - Array to accumulate detected pattern descriptions\n   * @param currentSeverity - Current highest severity level\n   * @param maxLength - Maximum allowed content length for regex validation\n   */\n  private static checkInjectionPatterns(\n    originalContent: string,\n    normalizedContent: string,\n    detectedPatterns: string[],\n    currentSeverity: SecuritySeverity,\n    maxLength: number,\n    contentContext?: ContentValidatorOptions['contentContext']\n  ): {\n    sanitized: string;\n    highestSeverity: SecuritySeverity;\n  } {\n    let sanitized = normalizedContent;\n    let highestSeverity = currentSeverity;\n\n    for (const { pattern, severity, description } of this.INJECTION_PATTERNS) {\n      // Fix #456/#1725: Skip code execution and security documentation patterns\n      // for element types that legitimately contain code and attack descriptions.\n      // Prompt injection, actual token exposure, and HTML/XSS remain active.\n      if (contentContext && this.CODE_EXEMPT_CONTEXTS.has(contentContext) &&\n          (this.CODE_EXECUTION_PATTERNS.has(description) || this.SECURITY_DOC_PATTERNS.has(description))) {\n        continue;\n      }\n      // Fix #803: Skip HTML section tag patterns for templates (use <script>/<style> as section delimiters)\n      if (contentContext === 'template' && this.HTML_SECTION_PATTERNS.has(description)) {\n        continue;\n      }\n      // Check pattern on original content (before normalization) to catch encoded attacks\n      if (RegexValidator.validate(originalContent, pattern, {\n        maxLength,\n        rejectDangerousPatterns: false,\n        logEvents: false\n      })) {\n        detectedPatterns.push(description);\n\n        // Update highest severity\n        if (severity === 'critical' || (severity === 'high' && highestSeverity !== 'critical')) {\n          highestSeverity = severity;\n        }\n\n        // Log security event\n        SecurityMonitor.logSecurityEvent({\n          type: 'CONTENT_INJECTION_ATTEMPT',\n          severity: severity.toUpperCase() as 'HIGH' | 'CRITICAL',\n          source: 'content_validation',\n          details: `Detected pattern: ${description}`,\n        });\n\n        // Record in telemetry\n        ContentValidator.getTelemetry()?.recordBlockedAttack(\n          'CONTENT_INJECTION',\n          description,\n          severity.toUpperCase() as 'HIGH' | 'CRITICAL',\n          'content_validation',\n          { pattern: pattern.source }\n        );\n\n        // Apply replacement to normalized content\n        sanitized = sanitized.replace(pattern, '[CONTENT_BLOCKED]');\n      }\n    }\n\n    return { sanitized, highestSeverity };\n  }\n\n  /**\n   * Validates and sanitizes persona content for security threats\n   * FIX #1269: Added options to support large memory content\n   * REFACTOR: Reduced cognitive complexity by extracting helper methods\n   *\n   * SECURITY FIX (DMCP-SEC-004): Length checks now performed on NORMALIZED content\n   * to prevent bypass attacks using Unicode combining characters or zero-width chars.\n   * A pre-check with generous multiplier prevents DoS from huge payloads.\n   */\n  static validateAndSanitize(content: string, options: ContentValidatorOptions = {}): ContentValidationResult {\n    // Determine max length for validation\n    const maxLength = options.maxLength || SECURITY_LIMITS.MAX_CONTENT_LENGTH;\n\n    // SECURITY FIX (DMCP-SEC-004): Two-phase length validation\n    // Phase 1: DoS prevention pre-check on raw content (generous 2x multiplier)\n    // This prevents huge payloads from hitting the normalization code path\n    // while still allowing legitimate content with some Unicode overhead\n    const DOS_PREVENTION_MULTIPLIER = 2;\n    if (!options.skipSizeCheck) {\n      if (content.length > maxLength * DOS_PREVENTION_MULTIPLIER) {\n        throw new SecurityError(\n          `Content exceeds maximum length of ${maxLength} characters (${content.length} provided)`\n        );\n      }\n    }\n\n    const detectedPatterns: string[] = [];\n\n    // Handle Unicode validation (normalizes content but doesn't abort)\n    const unicodeCheck = this.handleUnicodeValidation(content, detectedPatterns);\n\n    // SECURITY FIX (DMCP-SEC-004): Phase 2 - Check length on NORMALIZED content\n    // This prevents bypass attacks using combining characters or zero-width chars\n    // that would inflate raw length but collapse after normalization\n    if (!options.skipSizeCheck) {\n      if (unicodeCheck.sanitized.length > maxLength) {\n        throw new SecurityError(\n          `Content exceeds maximum length of ${maxLength} characters after normalization (${unicodeCheck.sanitized.length} provided)`\n        );\n      }\n    }\n\n    // Check for injection patterns on ORIGINAL content (to catch encoded attacks)\n    // but apply replacements to NORMALIZED content (to preserve normalization)\n    const injectionCheck = this.checkInjectionPatterns(\n      content,\n      unicodeCheck.sanitized,\n      detectedPatterns,\n      unicodeCheck.highestSeverity,\n      maxLength,\n      options.contentContext\n    );\n\n    // Use highest severity from either Unicode or injection checks\n    const finalSeverity = injectionCheck.highestSeverity;\n\n    // Abort if high/critical threats detected\n    if (finalSeverity === 'high' || finalSeverity === 'critical') {\n      return {\n        isValid: false,\n        sanitizedContent: injectionCheck.sanitized,\n        detectedPatterns,\n        severity: finalSeverity\n      };\n    }\n\n    return {\n      isValid: detectedPatterns.length === 0,\n      sanitizedContent: injectionCheck.sanitized,\n      detectedPatterns,\n      severity: finalSeverity\n    };\n  }\n\n  /**\n   * Validates YAML frontmatter for malicious content\n   * SECURITY FIX #364: Added YAML bomb detection to prevent denial of service\n   */\n  static validateYamlContent(yamlContent: string): boolean {\n    // Length validation before pattern matching\n    if (yamlContent.length > SECURITY_LIMITS.MAX_YAML_LENGTH) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'YAML_INJECTION_ATTEMPT',\n        severity: 'HIGH',\n        source: 'yaml_validation',\n        details: `YAML content exceeds maximum length: ${yamlContent.length} > ${SECURITY_LIMITS.MAX_YAML_LENGTH}`\n      });\n      return false;\n    }\n\n    // SECURITY FIX #364: Check for YAML bombs before other validation\n    // SECURITY FIX (PR #552 review): Use RegexValidator for ReDoS protection\n    for (const pattern of this.YAML_BOMB_PATTERNS) {\n      // Use RegexValidator to safely check patterns with timeout protection\n      // This prevents ReDoS attacks from maliciously crafted YAML\n      const isMatch = RegexValidator.validate(yamlContent, pattern, {\n        maxLength: SECURITY_LIMITS.MAX_YAML_LENGTH,\n        rejectDangerousPatterns: false, // Our patterns are trusted\n        logEvents: false // We handle logging ourselves\n      });\n      \n      if (isMatch) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'YAML_INJECTION_ATTEMPT',\n          severity: 'CRITICAL',\n          source: 'yaml_bomb_detection',\n          details: `YAML bomb pattern detected: ${pattern.source}`,\n          metadata: {\n            patternType: 'YAML_BOMB',\n            contentLength: yamlContent.length\n          }\n        });\n\n        // Record in telemetry\n        ContentValidator.getTelemetry()?.recordBlockedAttack(\n          'YAML_BOMB',\n          `YAML bomb pattern: ${pattern.source}`,\n          'CRITICAL',\n          'yaml_validation',\n          { patternType: 'YAML_BOMB', contentLength: yamlContent.length }\n        );\n\n        return false;\n      }\n    }\n    \n    // SECURITY FIX #364: Count anchor/alias ratio for amplification detection\n    // SECURITY FIX #1298: Use configurable threshold for easier tuning\n    const anchorMatches = yamlContent.match(/&\\w+/g) || [];\n    // Fix #906: Use negative lookbehind to exclude markdown bold (**word**) from\n    // matching as YAML aliases. Without this, markdown bold inside YAML strings\n    // triggers false-positive amplification detection.\n    const aliasMatches = yamlContent.match(/(?<!\\*)\\*\\w+/g) || [];\n    const amplificationRatio = anchorMatches.length > 0 ? aliasMatches.length / anchorMatches.length : 0;\n\n    if (amplificationRatio > SECURITY_LIMITS.YAML_BOMB_AMPLIFICATION_THRESHOLD) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'YAML_INJECTION_ATTEMPT',\n        severity: 'HIGH',\n        source: 'yaml_amplification_detection',\n        details: `Excessive alias amplification detected: ${aliasMatches.length} aliases for ${anchorMatches.length} anchors (ratio: ${amplificationRatio.toFixed(2)})`,\n        metadata: {\n          anchors: anchorMatches.length,\n          aliases: aliasMatches.length,\n          ratio: amplificationRatio\n        }\n      });\n      return false;\n    }\n    \n    // SECURITY FIX #364: Detect circular reference chains\n    // SECURITY FIX (PR #552 review): Optimized from O(n²) to O(n) using Set-based lookups\n    const anchorRefs = new Map<string, Set<string>>();\n    const lines = yamlContent.split('\\n');\n    \n    // First pass: Build reference map efficiently\n    for (let i = 0; i < lines.length; i++) {\n      const anchorMatch = lines[i].match(/&(\\w+)/);\n      if (anchorMatch) {\n        const anchorName = anchorMatch[1];\n        // Get references in next 5 lines\n        const contextEnd = Math.min(i + 5, lines.length);\n        const references = new Set<string>();\n        \n        for (let j = i; j < contextEnd; j++) {\n          const aliasMatches = lines[j].match(/\\*(\\w+)/g);\n          if (aliasMatches) {\n            aliasMatches.forEach(alias => {\n              references.add(alias.substring(1)); // Remove * prefix\n            });\n          }\n        }\n        \n        anchorRefs.set(anchorName, references);\n      }\n    }\n    \n    // Second pass: Check for circular references (O(n) with Set lookups)\n    for (const [anchor1, refs1] of anchorRefs) {\n      for (const refAnchor of refs1) {\n        const refs2 = anchorRefs.get(refAnchor);\n        // Check if the referenced anchor references back to the original\n        if (refs2 && refs2.has(anchor1)) {\n          SecurityMonitor.logSecurityEvent({\n            type: 'YAML_INJECTION_ATTEMPT',\n            severity: 'CRITICAL',\n            source: 'yaml_bomb_detection',\n            details: `Circular reference chain detected between anchors: &${anchor1} and &${refAnchor}`,\n            metadata: {\n              patternType: 'CIRCULAR_REFERENCE',\n              anchors: [anchor1, refAnchor]\n            }\n          });\n          return false;\n        }\n      }\n    }\n    \n    // Unicode normalization preprocessing for YAML content\n    const unicodeResult = UnicodeValidator.normalize(yamlContent);\n    const normalizedYaml = unicodeResult.normalizedContent;\n    \n    if (!unicodeResult.isValid && unicodeResult.detectedIssues) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'YAML_UNICODE_ATTACK',\n        severity: (unicodeResult.severity?.toUpperCase() || 'MEDIUM') as 'LOW' | 'MEDIUM' | 'HIGH' | 'CRITICAL',\n        source: 'yaml_validation',\n        details: `Unicode attack detected in YAML: ${unicodeResult.detectedIssues.join(', ')}`\n      });\n      return false;\n    }\n\n    for (const pattern of this.MALICIOUS_YAML_PATTERNS) {\n      // These are trusted internal patterns, so we disable ReDoS rejection\n      if (RegexValidator.validate(normalizedYaml, pattern, {\n        maxLength: SECURITY_LIMITS.MAX_CONTENT_LENGTH,\n        rejectDangerousPatterns: false,\n        logEvents: false  // Don't log our own security patterns as dangerous\n      })) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'YAML_INJECTION_ATTEMPT',\n          severity: 'CRITICAL',\n          source: 'yaml_validation',\n          details: `Malicious YAML pattern detected: ${pattern}`,\n        });\n        // Early exit on first match for performance\n        return false;\n      }\n    }\n    return true;\n  }\n\n  /**\n   * Validates persona metadata fields\n   */\n  static validateMetadata(metadata: any): ContentValidationResult {\n    const detectedPatterns: string[] = [];\n\n    // Check all string fields in metadata\n    const checkField = (fieldName: string, value: any) => {\n      if (typeof value === 'string') {\n        // Check field length first\n        if (value.length > SECURITY_LIMITS.MAX_METADATA_FIELD_LENGTH) {\n          detectedPatterns.push(`${fieldName}: Field exceeds maximum length of ${SECURITY_LIMITS.MAX_METADATA_FIELD_LENGTH} characters`);\n          return;\n        }\n        \n        const result = this.validateAndSanitize(value);\n        if (!result.isValid || result.detectedPatterns?.length) {\n          detectedPatterns.push(`${fieldName}: ${result.detectedPatterns?.join(', ')}`);\n        }\n      }\n    };\n\n    // Validate standard persona fields\n    checkField('name', metadata.name);\n    checkField('description', metadata.description);\n    checkField('category', metadata.category);\n    checkField('author', metadata.author);\n    \n    // Check any custom fields\n    for (const [key, value] of Object.entries(metadata)) {\n      if (!['name', 'description', 'category', 'author'].includes(key)) {\n        checkField(key, value);\n      }\n    }\n\n    return {\n      isValid: detectedPatterns.length === 0,\n      detectedPatterns,\n      severity: detectedPatterns.length > 0 ? 'high' : 'low'\n    };\n  }\n\n  /**\n   * Sanitizes a complete persona file (frontmatter + content)\n   */\n  static sanitizePersonaContent(content: string): string {\n    // Extract frontmatter\n    const frontmatterMatch = content.match(/^---\\n([\\s\\S]*?)\\n---/);\n    \n    if (!frontmatterMatch) {\n      // No frontmatter, just validate content\n      const result = this.validateAndSanitize(content);\n      if (!result.isValid && result.severity === 'critical') {\n        // FIX: Include specific patterns that triggered the rejection for debugging\n        const patterns = result.detectedPatterns?.join(', ') || 'unknown patterns';\n        throw new SecurityError(`Critical security threat detected in persona content: ${patterns}`);\n      }\n      return result.sanitizedContent || content;\n    }\n\n    const yamlContent = frontmatterMatch[1];\n    const markdownContent = content.substring(frontmatterMatch[0].length);\n\n    // Validate YAML\n    if (!this.validateYamlContent(yamlContent)) {\n      throw new SecurityError('Malicious YAML detected in persona frontmatter');\n    }\n\n    // Validate markdown content\n    const contentResult = this.validateAndSanitize(markdownContent);\n    if (!contentResult.isValid && contentResult.severity === 'critical') {\n      // FIX: Include specific patterns that triggered the rejection for debugging\n      const patterns = contentResult.detectedPatterns?.join(', ') || 'unknown patterns';\n      throw new SecurityError(`Critical security threat detected in persona content: ${patterns}`);\n    }\n\n    // Return sanitized content\n    return `---\\n${yamlContent}\\n---${contentResult.sanitizedContent || markdownContent}`;\n  }\n}"]}
@@ -1 +1 @@
1
- {"version":3,"file":"unicodeValidator.d.ts","sourceRoot":"","sources":["../../../src/security/validators/unicodeValidator.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAIH,MAAM,WAAW,uBAAuB;IACtC,OAAO,EAAE,OAAO,CAAC;IACjB,iBAAiB,EAAE,MAAM,CAAC;IAC1B,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;IAC1B,QAAQ,CAAC,EAAE,KAAK,GAAG,QAAQ,GAAG,MAAM,GAAG,UAAU,CAAC;CACnD;AAED,qBAAa,gBAAgB;IAC3B;;OAEG;IAEH;;;;;OAKG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,wBAAwB,CAAmC;IAEnF;;;;;OAKG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,gBAAgB,CAAyC;IAEjF;;;;;OAKG;IAEH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAwE;IAEnH;;;OAGG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CA2BxC;IAEH;;;OAGG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,eAAe,CASrC;IAEF;;OAEG;IACH,MAAM,CAAC,SAAS,CAAC,OAAO,EAAE,MAAM,GAAG,uBAAuB;IAwG1D;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,wBAAwB;IA2CvC;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,kBAAkB;IAcjC;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,kBAAkB;IAoBjC;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,gBAAgB;IAW/B;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,WAAW;IAI1B;;OAEG;IACH,MAAM,CAAC,wBAAwB,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO;IAQzD;;;OAGG;IACH,OAAO,CAAC,MAAM,CAAC,0BAA0B;IAKzC;;;OAGG;IACH,OAAO,CAAC,MAAM,CAAC,sBAAsB;IA2BrC;;OAEG;IACH,MAAM,CAAC,cAAc,CAAC,OAAO,EAAE,MAAM,EAAE,SAAS,GAAE,MAAY,GAAG,MAAM;CAWxE"}
1
+ {"version":3,"file":"unicodeValidator.d.ts","sourceRoot":"","sources":["../../../src/security/validators/unicodeValidator.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAIH,MAAM,WAAW,uBAAuB;IACtC,OAAO,EAAE,OAAO,CAAC;IACjB,iBAAiB,EAAE,MAAM,CAAC;IAC1B,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;IAC1B,QAAQ,CAAC,EAAE,KAAK,GAAG,QAAQ,GAAG,MAAM,GAAG,UAAU,CAAC;CACnD;AAED,qBAAa,gBAAgB;IAC3B;;OAEG;IAEH;;;;;OAKG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,wBAAwB,CAAmC;IAEnF;;;;;OAKG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,gBAAgB,CAAyC;IAEjF;;;;;OAKG;IAEH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAwE;IAEnH;;;OAGG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CA2BxC;IAEH;;;OAGG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,eAAe,CASrC;IAEF;;OAEG;IACH,MAAM,CAAC,SAAS,CAAC,OAAO,EAAE,MAAM,GAAG,uBAAuB;IAwG1D;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,wBAAwB;IA2CvC;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,kBAAkB;IAcjC;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,kBAAkB;IAqBjC;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,gBAAgB;IAW/B;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,WAAW;IAI1B;;OAEG;IACH,MAAM,CAAC,wBAAwB,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO;IAQzD;;;OAGG;IACH,OAAO,CAAC,MAAM,CAAC,0BAA0B;IAKzC;;;OAGG;IACH,OAAO,CAAC,MAAM,CAAC,sBAAsB;IA2BrC;;OAEG;IACH,MAAM,CAAC,cAAc,CAAC,OAAO,EAAE,MAAM,EAAE,SAAS,GAAE,MAAY,GAAG,MAAM;CAWxE"}
@@ -240,11 +240,12 @@ export class UnicodeValidator {
240
240
  }
241
241
  // Consider it suspicious if:
242
242
  // 1. More than 3 scripts are mixed (legitimate text rarely mixes >3 scripts)
243
- // 2. Content contains Latin + dangerous confusable scripts (Cyrillic/Greek - common attack pattern)
243
+ // 2. Content contains Latin + Cyrillic (homoglyph attack Cyrillic а/о/р look identical to Latin)
244
244
  // Note: Latin + CJK is common and legitimate (e.g., Chinese with English)
245
+ // Note: Latin + Greek is common and legitimate (e.g., α, β, γ, π, Σ, Δ in math/science)
245
246
  const isSuspicious = detectedScripts.length > 3 ||
246
247
  (detectedScripts.includes('LATIN') && detectedScripts.length > 1 &&
247
- (detectedScripts.includes('CYRILLIC') || detectedScripts.includes('GREEK')));
248
+ detectedScripts.includes('CYRILLIC'));
248
249
  return { isSuspicious, scripts: detectedScripts };
249
250
  }
250
251
  /**
@@ -323,4 +324,4 @@ export class UnicodeValidator {
323
324
  cleaned;
324
325
  }
325
326
  }
326
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"unicodeValidator.js","sourceRoot":"","sources":["../../../src/security/validators/unicodeValidator.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAEH,OAAO,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AASxD,MAAM,OAAO,gBAAgB;IAC3B;;OAEG;IAEH;;;;;OAKG;IACK,MAAM,CAAU,wBAAwB,GAAG,+BAA+B,CAAC;IAEnF;;;;;OAKG;IACK,MAAM,CAAU,gBAAgB,GAAG,qCAAqC,CAAC;IAEjF;;;;;OAKG;IACH,8GAA8G;IACtG,MAAM,CAAU,mBAAmB,GAAG,oEAAoE,CAAC,CAAC,gFAAgF;IAEpM;;;OAGG;IACK,MAAM,CAAU,mBAAmB,GAAwB,IAAI,GAAG,CAAC;QACzE,oBAAoB;QACpB,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAClF,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAClF,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAE1D,iBAAiB;QACjB,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAClF,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QACnF,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAClF,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAEpC,iDAAiD;QACjD,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC;QAChV,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC;QAEhV,qCAAqC;QACrC,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAE9C,2BAA2B;QAC3B,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QACtE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAE5D,uBAAuB;QACvB,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QACtT,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE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","sourcesContent":["/**\n * Unicode Validator for DollhouseMCP\n * \n * Prevents Unicode-based bypass attacks including:\n * - Homograph attacks (visually similar characters)\n * - Direction override attacks (RLO/LRO)\n * - Mixed script attacks\n * - Zero-width character injection\n * - Unicode normalization bypasses\n * \n * Security: SEC-001 - Unicode attack prevention\n */\n\nimport { SecurityMonitor } from '../securityMonitor.js';\n\nexport interface UnicodeValidationResult {\n  isValid: boolean;\n  normalizedContent: string;\n  detectedIssues?: string[];\n  severity?: 'low' | 'medium' | 'high' | 'critical';\n}\n\nexport class UnicodeValidator {\n  /**\n   * Unicode attack patterns and confusable characters\n   */\n  \n  /**\n   * Direction override characters that can hide or reverse text display\n   * @see https://unicode.org/reports/tr9/#Directional_Formatting_Characters\n   * U+202A-U+202E: Left/Right embedding and override marks (LRE, RLE, PDF, LRO, RLO)\n   * U+2066-U+2069: Isolate formatting characters (LRI, RLI, FSI, PDI)\n   */\n  private static readonly DIRECTION_OVERRIDE_CHARS = /[\\u202A-\\u202E\\u2066-\\u2069]/g;\n  \n  /**\n   * Zero-width and invisible formatting characters often used to hide payloads\n   * U+200B-U+200F: Zero-width spaces and directional marks\n   * U+2028-U+202F: Line/paragraph separators and formatting characters\n   * U+FEFF: Zero-width no-break space (Byte Order Mark)\n   */\n  private static readonly ZERO_WIDTH_CHARS = /[\\u200B-\\u200F\\u2028-\\u202F\\uFEFF]/g;\n  \n  /**\n   * Non-printable control characters that should not appear in normal text\n   * U+0000-U+0008, U+000B-U+000C, U+000E-U+001F: C0 control codes (except TAB, LF, CR)\n   * U+007F-U+009F: Delete and C1 control codes\n   * U+FFFE-U+FFFF: Non-characters that should never appear in valid text\n   */\n  // eslint-disable-next-line no-control-regex -- Intentionally matching control chars for security sanitization\n  private static readonly NON_PRINTABLE_CHARS = /[\\u0000-\\u0008\\u000B\\u000C\\u000E-\\u001F\\u007F-\\u009F\\uFFFE\\uFFFF]/g; // NOSONAR - Intentionally matching control characters for security sanitization\n  \n  /**\n   * Common homograph/confusable character mappings\n   * Maps visually similar Unicode characters to their ASCII equivalents\n   */\n  private static readonly CONFUSABLE_MAPPINGS: Map<string, string> = new Map([\n    // Cyrillic to Latin\n    ['а', 'a'], ['е', 'e'], ['о', 'o'], ['р', 'p'], ['с', 'c'], ['х', 'x'], ['у', 'y'],\n    ['А', 'A'], ['В', 'B'], ['Е', 'E'], ['К', 'K'], ['М', 'M'], ['Н', 'H'], ['О', 'O'], \n    ['Р', 'P'], ['С', 'C'], ['Т', 'T'], ['У', 'Y'], ['Х', 'X'],\n    \n    // Greek to Latin\n    ['α', 'a'], ['β', 'b'], ['γ', 'g'], ['δ', 'd'], ['ε', 'e'], ['ζ', 'z'], ['η', 'h'],\n    ['θ', 'th'], ['ι', 'i'], ['κ', 'k'], ['λ', 'l'], ['μ', 'm'], ['ν', 'n'], ['ξ', 'x'],\n    ['ο', 'o'], ['π', 'p'], ['ρ', 'r'], ['σ', 's'], ['τ', 't'], ['υ', 'u'], ['φ', 'f'],\n    ['χ', 'ch'], ['ψ', 'ps'], ['ω', 'w'],\n    \n    // Mathematical symbols to ASCII (various styles)\n    ['𝒂', 'a'], ['𝒃', 'b'], ['𝒄', 'c'], ['𝒅', 'd'], ['𝒆', 'e'], ['𝒇', 'f'], ['𝒈', 'g'], ['𝒉', 'h'], ['𝒊', 'i'], ['𝒋', 'j'], ['𝒌', 'k'], ['𝒍', 'l'], ['𝒎', 'm'], ['𝒏', 'n'], ['𝒐', 'o'], ['𝒑', 'p'], ['𝒒', 'q'], ['𝒓', 'r'], ['𝒔', 's'], ['𝒕', 't'], ['𝒖', 'u'], ['𝒗', 'v'], ['𝒘', 'w'], ['𝒙', 'x'], ['𝒚', 'y'], ['𝒛', 'z'],\n    ['𝐚', 'a'], ['𝐛', 'b'], ['𝐜', 'c'], ['𝐝', 'd'], ['𝐞', 'e'], ['𝐟', 'f'], ['𝐠', 'g'], ['𝐡', 'h'], ['𝐢', 'i'], ['𝐣', 'j'], ['𝐤', 'k'], ['𝐥', 'l'], ['𝐦', 'm'], ['𝐧', 'n'], ['𝐨', 'o'], ['𝐩', 'p'], ['𝐪', 'q'], ['𝐫', 'r'], ['𝐬', 's'], ['𝐭', 't'], ['𝐮', 'u'], ['𝐯', 'v'], ['𝐰', 'w'], ['𝐱', 'x'], ['𝐲', 'y'], ['𝐳', 'z'],\n    \n    // Special i variants (Turkish, etc.)\n    ['ı', 'i'], ['İ', 'I'], ['і', 'i'], ['Ӏ', 'I'],\n    \n    // Other common confusables\n    ['ǝ', 'e'], ['ɐ', 'a'], ['ɔ', 'o'], ['ʇ', 't'], ['ʌ', 'v'], ['ʍ', 'w'],\n    ['℃', 'C'], ['℉', 'F'], ['№', 'No'], ['™', 'TM'], ['®', 'R'],\n    \n    // Fullwidth characters\n    ['Ａ', 'A'], ['Ｂ', 'B'], ['Ｃ', 'C'], ['Ｄ', 'D'], ['Ｅ', 'E'], ['Ｆ', 'F'], ['Ｇ', 'G'], ['Ｈ', 'H'], ['Ｉ', 'I'], ['Ｊ', 'J'], ['Ｋ', 'K'], ['Ｌ', 'L'], ['Ｍ', 'M'], ['Ｎ', 'N'], ['Ｏ', 'O'], ['Ｐ', 'P'], ['Ｑ', 'Q'], ['Ｒ', 'R'], ['Ｓ', 'S'], ['Ｔ', 'T'], ['Ｕ', 'U'], ['Ｖ', 'V'], ['Ｗ', 'W'], ['Ｘ', 'X'], ['Ｙ', 'Y'], ['Ｚ', 'Z'],\n    ['ａ', 'a'], ['ｂ', 'b'], ['ｃ', 'c'], ['ｄ', 'd'], ['ｅ', 'e'], ['ｆ', 'f'], ['ｇ', 'g'], ['ｈ', 'h'], ['ｉ', 'i'], ['ｊ', 'j'], ['ｋ', 'k'], ['ｌ', 'l'], ['ｍ', 'm'], ['ｎ', 'n'], ['ｏ', 'o'], ['ｐ', 'p'], ['ｑ', 'q'], ['ｒ', 'r'], ['ｓ', 's'], ['ｔ', 't'], ['ｕ', 'u'], ['ｖ', 'v'], ['ｗ', 'w'], ['ｘ', 'x'], ['ｙ', 'y'], ['ｚ', 'z'],\n    ['０', '0'], ['１', '1'], ['２', '2'], ['３', '3'], ['４', '4'], ['５', '5'], ['６', '6'], ['７', '7'], ['８', '8'], ['９', '9'],\n  ]);\n\n  /**\n   * Script mixing detection patterns\n   * Detects suspicious mixing of different Unicode scripts\n   */\n  private static readonly SCRIPT_PATTERNS = {\n    // eslint-disable-next-line no-control-regex -- Intentionally includes control chars for comprehensive Latin script detection\n    LATIN: /[\\u0000-\\u007F\\u00A0-\\u00FF\\u0100-\\u017F\\u0180-\\u024F]/, // NOSONAR - Intentionally includes control characters for comprehensive Latin script detection\n    // Use alternation to avoid SonarCloud thinking \\u052F\\u2DE0 is a combined character\n    CYRILLIC: /(?:[\\u0400-\\u04FF]|[\\u0500-\\u052F]|[\\u2DE0-\\u2DFF]|[\\uA640-\\uA69F])/,\n    GREEK: /[\\u0370-\\u03FF\\u1F00-\\u1FFF]/,\n    ARABIC: /[\\u0600-\\u06FF\\u0750-\\u077F\\u08A0-\\u08FF\\uFB50-\\uFDFF\\uFE70-\\uFEFF]/,\n    HEBREW: /[\\u0590-\\u05FF\\uFB1D-\\uFB4F]/,\n    CJK: /[\\u2E80-\\u2EFF\\u2F00-\\u2FDF\\u3000-\\u303F\\u3040-\\u309F\\u30A0-\\u30FF\\u3100-\\u312F\\u3130-\\u318F\\u3190-\\u319F\\u31A0-\\u31BF\\u31C0-\\u31EF\\u31F0-\\u31FF\\u3200-\\u32FF\\u3300-\\u33FF\\u3400-\\u4DBF\\u4DC0-\\u4DFF\\u4E00-\\u9FFF]/,\n  };\n\n  /**\n   * Normalize Unicode content to prevent bypass attacks\n   */\n  static normalize(content: string): UnicodeValidationResult {\n    const issues: string[] = [];\n    let normalized = content;\n    let severity: 'low' | 'medium' | 'high' | 'critical' = 'low';\n\n    try {\n      // 1. Detect and log suspicious Unicode patterns before normalization\n      const suspiciousPatterns = this.detectSuspiciousPatterns(content);\n      issues.push(...suspiciousPatterns.issues);\n      if (suspiciousPatterns.severity) {\n        severity = this.escalateSeverity(severity, suspiciousPatterns.severity);\n      }\n\n      // 2. Remove direction override characters (prevents RLO/LRO attacks)\n      if (this.DIRECTION_OVERRIDE_CHARS.test(normalized)) {\n        issues.push('Direction override characters detected');\n        severity = this.escalateSeverity(severity, 'high');\n        normalized = normalized.replace(this.DIRECTION_OVERRIDE_CHARS, '');\n        \n        SecurityMonitor.logSecurityEvent({\n          type: 'UNICODE_DIRECTION_OVERRIDE',\n          severity: 'HIGH',\n          source: 'UnicodeValidator',\n          details: 'Direction override characters removed from content'\n        });\n      }\n\n      // 3. Remove zero-width and non-printable characters\n      if (this.ZERO_WIDTH_CHARS.test(normalized) || this.NON_PRINTABLE_CHARS.test(normalized)) {\n        // Check if the zero-width chars include direction marks (U+200E, U+200F)\n        const hasDirectionMarks = /[\\u200E\\u200F]/.test(normalized);\n        if (hasDirectionMarks) {\n          issues.push('Direction marks (LRM/RLM) detected');\n          severity = this.escalateSeverity(severity, 'high');\n        } else {\n          issues.push('Zero-width or non-printable characters detected');\n          severity = this.escalateSeverity(severity, 'medium');\n        }\n        normalized = normalized\n          .replace(this.ZERO_WIDTH_CHARS, '')\n          .replace(this.NON_PRINTABLE_CHARS, '');\n      }\n\n      // 4. Apply Unicode normalization (NFC - Canonical Decomposition + Composition)\n      normalized = normalized.normalize('NFC');\n\n      // 5. Detect mixed script attacks BEFORE confusable replacement\n      const mixedScriptResult = this.detectMixedScripts(normalized);\n      if (mixedScriptResult.isSuspicious) {\n        issues.push(`Mixed script usage detected: ${mixedScriptResult.scripts.join(', ')}`);\n        severity = this.escalateSeverity(severity, 'high');\n        \n        SecurityMonitor.logSecurityEvent({\n          type: 'UNICODE_MIXED_SCRIPT',\n          severity: 'HIGH',\n          source: 'UnicodeValidator',\n          details: `Mixed scripts detected: ${mixedScriptResult.scripts.join(', ')}`\n        });\n      }\n\n      // 6. Always replace confusable characters with ASCII equivalents for security\n      // This prevents homograph attacks regardless of script mixing\n      const confusableResult = this.replaceConfusables(normalized);\n      if (confusableResult.hasConfusables) {\n        normalized = confusableResult.normalized;\n        issues.push('Confusable Unicode characters detected and normalized');\n        severity = this.escalateSeverity(severity, 'medium');\n        \n        // Log if this happens in legitimate multilingual context\n        if (!mixedScriptResult.isSuspicious) {\n          SecurityMonitor.logSecurityEvent({\n            type: 'UNICODE_VALIDATION_ERROR',\n            severity: 'LOW',\n            source: 'UnicodeValidator',\n            details: 'Confusable characters normalized in legitimate multilingual content'\n          });\n        }\n      }\n\n      return {\n        isValid: issues.length === 0,\n        normalizedContent: normalized,\n        detectedIssues: issues.length > 0 ? issues : undefined,\n        severity: issues.length > 0 ? severity : undefined\n      };\n\n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'UNICODE_VALIDATION_ERROR',\n        severity: 'HIGH',\n        source: 'UnicodeValidator',\n        details: `Unicode validation failed: ${error instanceof Error ? error.message : String(error)}`\n      });\n\n      // Fallback: return original content if normalization fails\n      return {\n        isValid: false,\n        normalizedContent: content,\n        detectedIssues: ['Unicode validation failed'],\n        severity: 'high'\n      };\n    }\n  }\n\n  /**\n   * Detect suspicious Unicode patterns that might indicate attacks\n   */\n  private static detectSuspiciousPatterns(content: string): { issues: string[]; severity?: 'low' | 'medium' | 'high' | 'critical' } {\n    const issues: string[] = [];\n    let severity: 'low' | 'medium' | 'high' | 'critical' | undefined;\n\n    // Check for excessive Unicode escapes (possible encoding bypass)\n    /**\n     * Pattern to match Unicode escape sequences\n     * \\\\u: Literal backslash followed by 'u'\n     * [0-9a-fA-F]{4}: Exactly 4 hexadecimal digits\n     * Used to detect attempts to bypass filters using \\u0061dmin style encoding\n     */\n    const unicodeEscapePattern = /\\\\u[0-9a-fA-F]{4}/g;\n    const unicodeEscapes = content.match(unicodeEscapePattern);\n    if (unicodeEscapes && unicodeEscapes.length > 10) {\n      issues.push(`Excessive Unicode escapes detected (${unicodeEscapes.length})`);\n      severity = 'high';\n    }\n\n    // Check for suspicious Unicode ranges that might hide content\n    const suspiciousRanges = [\n      { range: /[\\uE000-\\uF8FF]/g, name: 'Private Use Area' },\n      // Note: Properly paired surrogate pairs [\\uD800-\\uDFFF] are normal for emojis\n      { range: /[\\uFDD0-\\uFDEF]/g, name: 'Non-characters' },\n      { range: /[\\uFFFE\\uFFFF]/g, name: 'Non-characters' }\n    ];\n\n    for (const { range, name } of suspiciousRanges) {\n      if (range.test(content)) {\n        issues.push(`Suspicious Unicode range detected: ${name}`);\n        severity = this.escalateSeverity(severity, 'medium');\n      }\n    }\n\n    // Check for malformed surrogate pairs using safe character-by-character validation\n    // This avoids ReDoS vulnerabilities from complex regex patterns\n    if (this.hasMalformedSurrogates(content)) {\n      issues.push('Malformed surrogate pairs detected');\n      severity = this.escalateSeverity(severity, 'high');\n    }\n\n    return { issues, severity };\n  }\n\n  /**\n   * Replace confusable Unicode characters with ASCII equivalents\n   */\n  private static replaceConfusables(content: string): { normalized: string; hasConfusables: boolean } {\n    let normalized = content;\n    let hasConfusables = false;\n\n    for (const [confusable, replacement] of this.CONFUSABLE_MAPPINGS) {\n      if (normalized.includes(confusable)) {\n        normalized = normalized.replace(new RegExp(this.escapeRegex(confusable), 'g'), replacement);\n        hasConfusables = true;\n      }\n    }\n\n    return { normalized, hasConfusables };\n  }\n\n  /**\n   * Detect suspicious mixing of different Unicode scripts\n   */\n  private static detectMixedScripts(content: string): { isSuspicious: boolean; scripts: string[] } {\n    const detectedScripts: string[] = [];\n\n    for (const [scriptName, pattern] of Object.entries(this.SCRIPT_PATTERNS)) {\n      if (pattern.test(content)) {\n        detectedScripts.push(scriptName);\n      }\n    }\n\n    // Consider it suspicious if:\n    // 1. More than 3 scripts are mixed (legitimate text rarely mixes >3 scripts)\n    // 2. Content contains Latin + dangerous confusable scripts (Cyrillic/Greek - common attack pattern)\n    // Note: Latin + CJK is common and legitimate (e.g., Chinese with English)\n    const isSuspicious = detectedScripts.length > 3 || \n      (detectedScripts.includes('LATIN') && detectedScripts.length > 1 && \n       (detectedScripts.includes('CYRILLIC') || detectedScripts.includes('GREEK')));\n\n    return { isSuspicious, scripts: detectedScripts };\n  }\n\n  /**\n   * Escalate severity level (higher severity takes precedence)\n   */\n  private static escalateSeverity(\n    current: 'low' | 'medium' | 'high' | 'critical' | undefined, \n    newSeverity: 'low' | 'medium' | 'high' | 'critical'\n  ): 'low' | 'medium' | 'high' | 'critical' {\n    const severityLevels = { low: 1, medium: 2, high: 3, critical: 4 };\n    const currentLevel = current ? severityLevels[current] : 0;\n    const newLevel = severityLevels[newSeverity];\n    \n    return newLevel > currentLevel ? newSeverity : (current || 'low');\n  }\n\n  /**\n   * Escape special regex characters for safe replacement\n   */\n  private static escapeRegex(string: string): string {\n    return string.replaceAll(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n  }\n\n  /**\n   * Check if content contains potentially dangerous Unicode patterns\n   */\n  static containsDangerousUnicode(content: string): boolean {\n    // Quick check for obviously dangerous patterns\n    return this.DIRECTION_OVERRIDE_CHARS.test(content) ||\n           this.ZERO_WIDTH_CHARS.test(content) ||\n           this.NON_PRINTABLE_CHARS.test(content) ||\n           this.hasExcessiveUnicodeEscapes(content);\n  }\n\n  /**\n   * Check if content has excessive Unicode escape sequences\n   * Prevents null pointer exception by safely checking match results\n   */\n  private static hasExcessiveUnicodeEscapes(content: string): boolean {\n    const matches = content.match(/\\\\u[0-9a-fA-F]{4}/g);\n    return matches !== null && matches.length > 10;\n  }\n\n  /**\n   * Safely check for malformed surrogate pairs without ReDoS vulnerability\n   * Uses character-by-character validation instead of complex regex\n   */\n  private static hasMalformedSurrogates(content: string): boolean {\n    for (let i = 0; i < content.length; i++) {\n      // SONARCLOUD FALSE POSITIVE (S7758): Must use charCodeAt here, not codePointAt\n      // This code specifically checks for malformed surrogate pairs at the 16-bit code unit level.\n      // codePointAt() would automatically combine valid pairs, making malformed detection impossible.\n      const char = content.charCodeAt(i);\n\n      // High surrogate (U+D800-U+DBFF)\n      if (char >= 0xD800 && char <= 0xDBFF) {\n        // Check if it's followed by a low surrogate\n        if (i + 1 >= content.length) {\n          return true; // High surrogate at end of string\n        }\n        const nextChar = content.charCodeAt(i + 1);\n        if (nextChar < 0xDC00 || nextChar > 0xDFFF) {\n          return true; // High surrogate not followed by low surrogate\n        }\n        i++; // Skip the valid low surrogate\n      }\n      // Low surrogate (U+DC00-U+DFFF) without preceding high surrogate\n      else if (char >= 0xDC00 && char <= 0xDFFF) {\n        return true; // Unpaired low surrogate\n      }\n    }\n    return false;\n  }\n\n  /**\n   * Get safe preview of Unicode content for logging\n   */\n  static getSafePreview(content: string, maxLength: number = 100): string {\n    // Remove dangerous Unicode characters and truncate for safe logging\n    const cleaned = content\n      .replace(this.DIRECTION_OVERRIDE_CHARS, '[DIR]')\n      .replace(this.ZERO_WIDTH_CHARS, '[ZW]')\n      .replace(this.NON_PRINTABLE_CHARS, '[NP]');\n    \n    return cleaned.length > maxLength ? \n      cleaned.substring(0, maxLength) + '...' : \n      cleaned;\n  }\n}\n"]}
327
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"unicodeValidator.js","sourceRoot":"","sources":["../../../src/security/validators/unicodeValidator.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAEH,OAAO,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AASxD,MAAM,OAAO,gBAAgB;IAC3B;;OAEG;IAEH;;;;;OAKG;IACK,MAAM,CAAU,wBAAwB,GAAG,+BAA+B,CAAC;IAEnF;;;;;OAKG;IACK,MAAM,CAAU,gBAAgB,GAAG,qCAAqC,CAAC;IAEjF;;;;;OAKG;IACH,8GAA8G;IACtG,MAAM,CAAU,mBAAmB,GAAG,oEAAoE,CAAC,CAAC,gFAAgF;IAEpM;;;OAGG;IACK,MAAM,CAAU,mBAAmB,GAAwB,IAAI,GAAG,CAAC;QACzE,oBAAoB;QACpB,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAClF,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAClF,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAE1D,iBAAiB;QACjB,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAClF,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QACnF,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAClF,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAEpC,iDAAiD;QACjD,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC;QAChV,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,CAAC;QAEhV,qCAAqC;QACrC,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAE9C,2BAA2B;QAC3B,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QACtE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QAE5D,uBAAuB;QACvB,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;QACtT,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE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","sourcesContent":["/**\n * Unicode Validator for DollhouseMCP\n * \n * Prevents Unicode-based bypass attacks including:\n * - Homograph attacks (visually similar characters)\n * - Direction override attacks (RLO/LRO)\n * - Mixed script attacks\n * - Zero-width character injection\n * - Unicode normalization bypasses\n * \n * Security: SEC-001 - Unicode attack prevention\n */\n\nimport { SecurityMonitor } from '../securityMonitor.js';\n\nexport interface UnicodeValidationResult {\n  isValid: boolean;\n  normalizedContent: string;\n  detectedIssues?: string[];\n  severity?: 'low' | 'medium' | 'high' | 'critical';\n}\n\nexport class UnicodeValidator {\n  /**\n   * Unicode attack patterns and confusable characters\n   */\n  \n  /**\n   * Direction override characters that can hide or reverse text display\n   * @see https://unicode.org/reports/tr9/#Directional_Formatting_Characters\n   * U+202A-U+202E: Left/Right embedding and override marks (LRE, RLE, PDF, LRO, RLO)\n   * U+2066-U+2069: Isolate formatting characters (LRI, RLI, FSI, PDI)\n   */\n  private static readonly DIRECTION_OVERRIDE_CHARS = /[\\u202A-\\u202E\\u2066-\\u2069]/g;\n  \n  /**\n   * Zero-width and invisible formatting characters often used to hide payloads\n   * U+200B-U+200F: Zero-width spaces and directional marks\n   * U+2028-U+202F: Line/paragraph separators and formatting characters\n   * U+FEFF: Zero-width no-break space (Byte Order Mark)\n   */\n  private static readonly ZERO_WIDTH_CHARS = /[\\u200B-\\u200F\\u2028-\\u202F\\uFEFF]/g;\n  \n  /**\n   * Non-printable control characters that should not appear in normal text\n   * U+0000-U+0008, U+000B-U+000C, U+000E-U+001F: C0 control codes (except TAB, LF, CR)\n   * U+007F-U+009F: Delete and C1 control codes\n   * U+FFFE-U+FFFF: Non-characters that should never appear in valid text\n   */\n  // eslint-disable-next-line no-control-regex -- Intentionally matching control chars for security sanitization\n  private static readonly NON_PRINTABLE_CHARS = /[\\u0000-\\u0008\\u000B\\u000C\\u000E-\\u001F\\u007F-\\u009F\\uFFFE\\uFFFF]/g; // NOSONAR - Intentionally matching control characters for security sanitization\n  \n  /**\n   * Common homograph/confusable character mappings\n   * Maps visually similar Unicode characters to their ASCII equivalents\n   */\n  private static readonly CONFUSABLE_MAPPINGS: Map<string, string> = new Map([\n    // Cyrillic to Latin\n    ['а', 'a'], ['е', 'e'], ['о', 'o'], ['р', 'p'], ['с', 'c'], ['х', 'x'], ['у', 'y'],\n    ['А', 'A'], ['В', 'B'], ['Е', 'E'], ['К', 'K'], ['М', 'M'], ['Н', 'H'], ['О', 'O'], \n    ['Р', 'P'], ['С', 'C'], ['Т', 'T'], ['У', 'Y'], ['Х', 'X'],\n    \n    // Greek to Latin\n    ['α', 'a'], ['β', 'b'], ['γ', 'g'], ['δ', 'd'], ['ε', 'e'], ['ζ', 'z'], ['η', 'h'],\n    ['θ', 'th'], ['ι', 'i'], ['κ', 'k'], ['λ', 'l'], ['μ', 'm'], ['ν', 'n'], ['ξ', 'x'],\n    ['ο', 'o'], ['π', 'p'], ['ρ', 'r'], ['σ', 's'], ['τ', 't'], ['υ', 'u'], ['φ', 'f'],\n    ['χ', 'ch'], ['ψ', 'ps'], ['ω', 'w'],\n    \n    // Mathematical symbols to ASCII (various styles)\n    ['𝒂', 'a'], ['𝒃', 'b'], ['𝒄', 'c'], ['𝒅', 'd'], ['𝒆', 'e'], ['𝒇', 'f'], ['𝒈', 'g'], ['𝒉', 'h'], ['𝒊', 'i'], ['𝒋', 'j'], ['𝒌', 'k'], ['𝒍', 'l'], ['𝒎', 'm'], ['𝒏', 'n'], ['𝒐', 'o'], ['𝒑', 'p'], ['𝒒', 'q'], ['𝒓', 'r'], ['𝒔', 's'], ['𝒕', 't'], ['𝒖', 'u'], ['𝒗', 'v'], ['𝒘', 'w'], ['𝒙', 'x'], ['𝒚', 'y'], ['𝒛', 'z'],\n    ['𝐚', 'a'], ['𝐛', 'b'], ['𝐜', 'c'], ['𝐝', 'd'], ['𝐞', 'e'], ['𝐟', 'f'], ['𝐠', 'g'], ['𝐡', 'h'], ['𝐢', 'i'], ['𝐣', 'j'], ['𝐤', 'k'], ['𝐥', 'l'], ['𝐦', 'm'], ['𝐧', 'n'], ['𝐨', 'o'], ['𝐩', 'p'], ['𝐪', 'q'], ['𝐫', 'r'], ['𝐬', 's'], ['𝐭', 't'], ['𝐮', 'u'], ['𝐯', 'v'], ['𝐰', 'w'], ['𝐱', 'x'], ['𝐲', 'y'], ['𝐳', 'z'],\n    \n    // Special i variants (Turkish, etc.)\n    ['ı', 'i'], ['İ', 'I'], ['і', 'i'], ['Ӏ', 'I'],\n    \n    // Other common confusables\n    ['ǝ', 'e'], ['ɐ', 'a'], ['ɔ', 'o'], ['ʇ', 't'], ['ʌ', 'v'], ['ʍ', 'w'],\n    ['℃', 'C'], ['℉', 'F'], ['№', 'No'], ['™', 'TM'], ['®', 'R'],\n    \n    // Fullwidth characters\n    ['Ａ', 'A'], ['Ｂ', 'B'], ['Ｃ', 'C'], ['Ｄ', 'D'], ['Ｅ', 'E'], ['Ｆ', 'F'], ['Ｇ', 'G'], ['Ｈ', 'H'], ['Ｉ', 'I'], ['Ｊ', 'J'], ['Ｋ', 'K'], ['Ｌ', 'L'], ['Ｍ', 'M'], ['Ｎ', 'N'], ['Ｏ', 'O'], ['Ｐ', 'P'], ['Ｑ', 'Q'], ['Ｒ', 'R'], ['Ｓ', 'S'], ['Ｔ', 'T'], ['Ｕ', 'U'], ['Ｖ', 'V'], ['Ｗ', 'W'], ['Ｘ', 'X'], ['Ｙ', 'Y'], ['Ｚ', 'Z'],\n    ['ａ', 'a'], ['ｂ', 'b'], ['ｃ', 'c'], ['ｄ', 'd'], ['ｅ', 'e'], ['ｆ', 'f'], ['ｇ', 'g'], ['ｈ', 'h'], ['ｉ', 'i'], ['ｊ', 'j'], ['ｋ', 'k'], ['ｌ', 'l'], ['ｍ', 'm'], ['ｎ', 'n'], ['ｏ', 'o'], ['ｐ', 'p'], ['ｑ', 'q'], ['ｒ', 'r'], ['ｓ', 's'], ['ｔ', 't'], ['ｕ', 'u'], ['ｖ', 'v'], ['ｗ', 'w'], ['ｘ', 'x'], ['ｙ', 'y'], ['ｚ', 'z'],\n    ['０', '0'], ['１', '1'], ['２', '2'], ['３', '3'], ['４', '4'], ['５', '5'], ['６', '6'], ['７', '7'], ['８', '8'], ['９', '9'],\n  ]);\n\n  /**\n   * Script mixing detection patterns\n   * Detects suspicious mixing of different Unicode scripts\n   */\n  private static readonly SCRIPT_PATTERNS = {\n    // eslint-disable-next-line no-control-regex -- Intentionally includes control chars for comprehensive Latin script detection\n    LATIN: /[\\u0000-\\u007F\\u00A0-\\u00FF\\u0100-\\u017F\\u0180-\\u024F]/, // NOSONAR - Intentionally includes control characters for comprehensive Latin script detection\n    // Use alternation to avoid SonarCloud thinking \\u052F\\u2DE0 is a combined character\n    CYRILLIC: /(?:[\\u0400-\\u04FF]|[\\u0500-\\u052F]|[\\u2DE0-\\u2DFF]|[\\uA640-\\uA69F])/,\n    GREEK: /[\\u0370-\\u03FF\\u1F00-\\u1FFF]/,\n    ARABIC: /[\\u0600-\\u06FF\\u0750-\\u077F\\u08A0-\\u08FF\\uFB50-\\uFDFF\\uFE70-\\uFEFF]/,\n    HEBREW: /[\\u0590-\\u05FF\\uFB1D-\\uFB4F]/,\n    CJK: /[\\u2E80-\\u2EFF\\u2F00-\\u2FDF\\u3000-\\u303F\\u3040-\\u309F\\u30A0-\\u30FF\\u3100-\\u312F\\u3130-\\u318F\\u3190-\\u319F\\u31A0-\\u31BF\\u31C0-\\u31EF\\u31F0-\\u31FF\\u3200-\\u32FF\\u3300-\\u33FF\\u3400-\\u4DBF\\u4DC0-\\u4DFF\\u4E00-\\u9FFF]/,\n  };\n\n  /**\n   * Normalize Unicode content to prevent bypass attacks\n   */\n  static normalize(content: string): UnicodeValidationResult {\n    const issues: string[] = [];\n    let normalized = content;\n    let severity: 'low' | 'medium' | 'high' | 'critical' = 'low';\n\n    try {\n      // 1. Detect and log suspicious Unicode patterns before normalization\n      const suspiciousPatterns = this.detectSuspiciousPatterns(content);\n      issues.push(...suspiciousPatterns.issues);\n      if (suspiciousPatterns.severity) {\n        severity = this.escalateSeverity(severity, suspiciousPatterns.severity);\n      }\n\n      // 2. Remove direction override characters (prevents RLO/LRO attacks)\n      if (this.DIRECTION_OVERRIDE_CHARS.test(normalized)) {\n        issues.push('Direction override characters detected');\n        severity = this.escalateSeverity(severity, 'high');\n        normalized = normalized.replace(this.DIRECTION_OVERRIDE_CHARS, '');\n        \n        SecurityMonitor.logSecurityEvent({\n          type: 'UNICODE_DIRECTION_OVERRIDE',\n          severity: 'HIGH',\n          source: 'UnicodeValidator',\n          details: 'Direction override characters removed from content'\n        });\n      }\n\n      // 3. Remove zero-width and non-printable characters\n      if (this.ZERO_WIDTH_CHARS.test(normalized) || this.NON_PRINTABLE_CHARS.test(normalized)) {\n        // Check if the zero-width chars include direction marks (U+200E, U+200F)\n        const hasDirectionMarks = /[\\u200E\\u200F]/.test(normalized);\n        if (hasDirectionMarks) {\n          issues.push('Direction marks (LRM/RLM) detected');\n          severity = this.escalateSeverity(severity, 'high');\n        } else {\n          issues.push('Zero-width or non-printable characters detected');\n          severity = this.escalateSeverity(severity, 'medium');\n        }\n        normalized = normalized\n          .replace(this.ZERO_WIDTH_CHARS, '')\n          .replace(this.NON_PRINTABLE_CHARS, '');\n      }\n\n      // 4. Apply Unicode normalization (NFC - Canonical Decomposition + Composition)\n      normalized = normalized.normalize('NFC');\n\n      // 5. Detect mixed script attacks BEFORE confusable replacement\n      const mixedScriptResult = this.detectMixedScripts(normalized);\n      if (mixedScriptResult.isSuspicious) {\n        issues.push(`Mixed script usage detected: ${mixedScriptResult.scripts.join(', ')}`);\n        severity = this.escalateSeverity(severity, 'high');\n        \n        SecurityMonitor.logSecurityEvent({\n          type: 'UNICODE_MIXED_SCRIPT',\n          severity: 'HIGH',\n          source: 'UnicodeValidator',\n          details: `Mixed scripts detected: ${mixedScriptResult.scripts.join(', ')}`\n        });\n      }\n\n      // 6. Always replace confusable characters with ASCII equivalents for security\n      // This prevents homograph attacks regardless of script mixing\n      const confusableResult = this.replaceConfusables(normalized);\n      if (confusableResult.hasConfusables) {\n        normalized = confusableResult.normalized;\n        issues.push('Confusable Unicode characters detected and normalized');\n        severity = this.escalateSeverity(severity, 'medium');\n        \n        // Log if this happens in legitimate multilingual context\n        if (!mixedScriptResult.isSuspicious) {\n          SecurityMonitor.logSecurityEvent({\n            type: 'UNICODE_VALIDATION_ERROR',\n            severity: 'LOW',\n            source: 'UnicodeValidator',\n            details: 'Confusable characters normalized in legitimate multilingual content'\n          });\n        }\n      }\n\n      return {\n        isValid: issues.length === 0,\n        normalizedContent: normalized,\n        detectedIssues: issues.length > 0 ? issues : undefined,\n        severity: issues.length > 0 ? severity : undefined\n      };\n\n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'UNICODE_VALIDATION_ERROR',\n        severity: 'HIGH',\n        source: 'UnicodeValidator',\n        details: `Unicode validation failed: ${error instanceof Error ? error.message : String(error)}`\n      });\n\n      // Fallback: return original content if normalization fails\n      return {\n        isValid: false,\n        normalizedContent: content,\n        detectedIssues: ['Unicode validation failed'],\n        severity: 'high'\n      };\n    }\n  }\n\n  /**\n   * Detect suspicious Unicode patterns that might indicate attacks\n   */\n  private static detectSuspiciousPatterns(content: string): { issues: string[]; severity?: 'low' | 'medium' | 'high' | 'critical' } {\n    const issues: string[] = [];\n    let severity: 'low' | 'medium' | 'high' | 'critical' | undefined;\n\n    // Check for excessive Unicode escapes (possible encoding bypass)\n    /**\n     * Pattern to match Unicode escape sequences\n     * \\\\u: Literal backslash followed by 'u'\n     * [0-9a-fA-F]{4}: Exactly 4 hexadecimal digits\n     * Used to detect attempts to bypass filters using \\u0061dmin style encoding\n     */\n    const unicodeEscapePattern = /\\\\u[0-9a-fA-F]{4}/g;\n    const unicodeEscapes = content.match(unicodeEscapePattern);\n    if (unicodeEscapes && unicodeEscapes.length > 10) {\n      issues.push(`Excessive Unicode escapes detected (${unicodeEscapes.length})`);\n      severity = 'high';\n    }\n\n    // Check for suspicious Unicode ranges that might hide content\n    const suspiciousRanges = [\n      { range: /[\\uE000-\\uF8FF]/g, name: 'Private Use Area' },\n      // Note: Properly paired surrogate pairs [\\uD800-\\uDFFF] are normal for emojis\n      { range: /[\\uFDD0-\\uFDEF]/g, name: 'Non-characters' },\n      { range: /[\\uFFFE\\uFFFF]/g, name: 'Non-characters' }\n    ];\n\n    for (const { range, name } of suspiciousRanges) {\n      if (range.test(content)) {\n        issues.push(`Suspicious Unicode range detected: ${name}`);\n        severity = this.escalateSeverity(severity, 'medium');\n      }\n    }\n\n    // Check for malformed surrogate pairs using safe character-by-character validation\n    // This avoids ReDoS vulnerabilities from complex regex patterns\n    if (this.hasMalformedSurrogates(content)) {\n      issues.push('Malformed surrogate pairs detected');\n      severity = this.escalateSeverity(severity, 'high');\n    }\n\n    return { issues, severity };\n  }\n\n  /**\n   * Replace confusable Unicode characters with ASCII equivalents\n   */\n  private static replaceConfusables(content: string): { normalized: string; hasConfusables: boolean } {\n    let normalized = content;\n    let hasConfusables = false;\n\n    for (const [confusable, replacement] of this.CONFUSABLE_MAPPINGS) {\n      if (normalized.includes(confusable)) {\n        normalized = normalized.replace(new RegExp(this.escapeRegex(confusable), 'g'), replacement);\n        hasConfusables = true;\n      }\n    }\n\n    return { normalized, hasConfusables };\n  }\n\n  /**\n   * Detect suspicious mixing of different Unicode scripts\n   */\n  private static detectMixedScripts(content: string): { isSuspicious: boolean; scripts: string[] } {\n    const detectedScripts: string[] = [];\n\n    for (const [scriptName, pattern] of Object.entries(this.SCRIPT_PATTERNS)) {\n      if (pattern.test(content)) {\n        detectedScripts.push(scriptName);\n      }\n    }\n\n    // Consider it suspicious if:\n    // 1. More than 3 scripts are mixed (legitimate text rarely mixes >3 scripts)\n    // 2. Content contains Latin + Cyrillic (homoglyph attack — Cyrillic а/о/р look identical to Latin)\n    // Note: Latin + CJK is common and legitimate (e.g., Chinese with English)\n    // Note: Latin + Greek is common and legitimate (e.g., α, β, γ, π, Σ, Δ in math/science)\n    const isSuspicious = detectedScripts.length > 3 ||\n      (detectedScripts.includes('LATIN') && detectedScripts.length > 1 &&\n       detectedScripts.includes('CYRILLIC'));\n\n    return { isSuspicious, scripts: detectedScripts };\n  }\n\n  /**\n   * Escalate severity level (higher severity takes precedence)\n   */\n  private static escalateSeverity(\n    current: 'low' | 'medium' | 'high' | 'critical' | undefined, \n    newSeverity: 'low' | 'medium' | 'high' | 'critical'\n  ): 'low' | 'medium' | 'high' | 'critical' {\n    const severityLevels = { low: 1, medium: 2, high: 3, critical: 4 };\n    const currentLevel = current ? severityLevels[current] : 0;\n    const newLevel = severityLevels[newSeverity];\n    \n    return newLevel > currentLevel ? newSeverity : (current || 'low');\n  }\n\n  /**\n   * Escape special regex characters for safe replacement\n   */\n  private static escapeRegex(string: string): string {\n    return string.replaceAll(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n  }\n\n  /**\n   * Check if content contains potentially dangerous Unicode patterns\n   */\n  static containsDangerousUnicode(content: string): boolean {\n    // Quick check for obviously dangerous patterns\n    return this.DIRECTION_OVERRIDE_CHARS.test(content) ||\n           this.ZERO_WIDTH_CHARS.test(content) ||\n           this.NON_PRINTABLE_CHARS.test(content) ||\n           this.hasExcessiveUnicodeEscapes(content);\n  }\n\n  /**\n   * Check if content has excessive Unicode escape sequences\n   * Prevents null pointer exception by safely checking match results\n   */\n  private static hasExcessiveUnicodeEscapes(content: string): boolean {\n    const matches = content.match(/\\\\u[0-9a-fA-F]{4}/g);\n    return matches !== null && matches.length > 10;\n  }\n\n  /**\n   * Safely check for malformed surrogate pairs without ReDoS vulnerability\n   * Uses character-by-character validation instead of complex regex\n   */\n  private static hasMalformedSurrogates(content: string): boolean {\n    for (let i = 0; i < content.length; i++) {\n      // SONARCLOUD FALSE POSITIVE (S7758): Must use charCodeAt here, not codePointAt\n      // This code specifically checks for malformed surrogate pairs at the 16-bit code unit level.\n      // codePointAt() would automatically combine valid pairs, making malformed detection impossible.\n      const char = content.charCodeAt(i);\n\n      // High surrogate (U+D800-U+DBFF)\n      if (char >= 0xD800 && char <= 0xDBFF) {\n        // Check if it's followed by a low surrogate\n        if (i + 1 >= content.length) {\n          return true; // High surrogate at end of string\n        }\n        const nextChar = content.charCodeAt(i + 1);\n        if (nextChar < 0xDC00 || nextChar > 0xDFFF) {\n          return true; // High surrogate not followed by low surrogate\n        }\n        i++; // Skip the valid low surrogate\n      }\n      // Low surrogate (U+DC00-U+DFFF) without preceding high surrogate\n      else if (char >= 0xDC00 && char <= 0xDFFF) {\n        return true; // Unpaired low surrogate\n      }\n    }\n    return false;\n  }\n\n  /**\n   * Get safe preview of Unicode content for logging\n   */\n  static getSafePreview(content: string, maxLength: number = 100): string {\n    // Remove dangerous Unicode characters and truncate for safe logging\n    const cleaned = content\n      .replace(this.DIRECTION_OVERRIDE_CHARS, '[DIR]')\n      .replace(this.ZERO_WIDTH_CHARS, '[ZW]')\n      .replace(this.NON_PRINTABLE_CHARS, '[NP]');\n    \n    return cleaned.length > maxLength ? \n      cleaned.substring(0, maxLength) + '...' : \n      cleaned;\n  }\n}\n"]}