@dollhousemcp/mcp-server 2.0.7 → 2.0.9
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/security/contentValidator.d.ts +20 -2
- package/dist/security/contentValidator.d.ts.map +1 -1
- package/dist/security/contentValidator.js +40 -5
- package/dist/security/validators/unicodeValidator.d.ts.map +1 -1
- package/dist/security/validators/unicodeValidator.js +4 -3
- package/dist/web/server.d.ts.map +1 -1
- package/dist/web/server.js +23 -10
- package/package.json +2 -2
- package/server.json +2 -2
|
@@ -2,8 +2,8 @@
|
|
|
2
2
|
* Auto-generated file - DO NOT EDIT
|
|
3
3
|
* Generated at build time by scripts/generate-version.js
|
|
4
4
|
*/
|
|
5
|
-
export declare const PACKAGE_VERSION = "2.0.
|
|
6
|
-
export declare const BUILD_TIMESTAMP = "2026-04-
|
|
5
|
+
export declare const PACKAGE_VERSION = "2.0.9";
|
|
6
|
+
export declare const BUILD_TIMESTAMP = "2026-04-03T16:02:10.270Z";
|
|
7
7
|
export declare const BUILD_TYPE: 'npm' | 'git';
|
|
8
8
|
export declare const PACKAGE_NAME = "@dollhousemcp/mcp-server";
|
|
9
9
|
//# sourceMappingURL=version.d.ts.map
|
|
@@ -2,8 +2,8 @@
|
|
|
2
2
|
* Auto-generated file - DO NOT EDIT
|
|
3
3
|
* Generated at build time by scripts/generate-version.js
|
|
4
4
|
*/
|
|
5
|
-
export const PACKAGE_VERSION = '2.0.
|
|
6
|
-
export const BUILD_TIMESTAMP = '2026-04-
|
|
5
|
+
export const PACKAGE_VERSION = '2.0.9';
|
|
6
|
+
export const BUILD_TIMESTAMP = '2026-04-03T16:02:10.270Z';
|
|
7
7
|
export const BUILD_TYPE = 'npm';
|
|
8
8
|
export const PACKAGE_NAME = '@dollhousemcp/mcp-server';
|
|
9
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuOSc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjYtMDQtMDNUMTY6MDI6MTAuMjcwWic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
|
|
@@ -56,8 +56,9 @@ export declare class ContentValidator {
|
|
|
56
56
|
* Content contexts where code execution patterns are legitimate and should
|
|
57
57
|
* not trigger security blocks. Skills contain exemplar code; templates contain
|
|
58
58
|
* code snippets that are rendered, never executed; agent definitions describe
|
|
59
|
-
* technical workflows that may reference code. Prompt injection,
|
|
60
|
-
*
|
|
59
|
+
* technical workflows that may reference code. Prompt injection, actual token
|
|
60
|
+
* exposure (ghp_/gho_), data exfiltration, and HTML/XSS patterns remain
|
|
61
|
+
* active for ALL contexts.
|
|
61
62
|
* @since Issue #456
|
|
62
63
|
*/
|
|
63
64
|
private static readonly CODE_EXEMPT_CONTEXTS;
|
|
@@ -67,6 +68,23 @@ export declare class ContentValidator {
|
|
|
67
68
|
* @since Issue #456
|
|
68
69
|
*/
|
|
69
70
|
private static readonly CODE_EXECUTION_PATTERNS;
|
|
71
|
+
/**
|
|
72
|
+
* Security documentation patterns exempt for CODE_EXEMPT_CONTEXTS.
|
|
73
|
+
* Skills/agents that teach penetration testing, threat modeling, etc.
|
|
74
|
+
* legitimately reference shell commands, file paths, and credential names.
|
|
75
|
+
*
|
|
76
|
+
* DISTINCTION FROM ACTIVE THREAT PATTERNS:
|
|
77
|
+
* These patterns describe attacks (educational) — they appear in element
|
|
78
|
+
* definitions that an author wrote, not in runtime user input. Patterns
|
|
79
|
+
* that remain active even in exempt contexts are actual threats:
|
|
80
|
+
* - Prompt injection (system/admin override, instruction manipulation)
|
|
81
|
+
* - Real token formats (ghp_*, gho_* — not just the word "GITHUB_TOKEN")
|
|
82
|
+
* - Data exfiltration commands (export/send all credentials)
|
|
83
|
+
* - HTML/XSS injection (renders in the web console)
|
|
84
|
+
*
|
|
85
|
+
* @since Issue #1725
|
|
86
|
+
*/
|
|
87
|
+
private static readonly SECURITY_DOC_PATTERNS;
|
|
70
88
|
/**
|
|
71
89
|
* HTML/XSS pattern descriptions exempt for template context.
|
|
72
90
|
* Templates use <template>, <style>, <script> as section delimiters.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"contentValidator.d.ts","sourceRoot":"","sources":["../../src/security/contentValidator.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAOH,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,kCAAkC,CAAC;AAG1E,MAAM,MAAM,gBAAgB,GAAG,KAAK,GAAG,QAAQ,GAAG,MAAM,GAAG,UAAU,CAAC;AAEtE,MAAM,WAAW,uBAAuB;IACtC,OAAO,EAAE,OAAO,CAAC;IACjB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,gBAAgB,CAAC,EAAE,MAAM,EAAE,CAAC;IAC5B,QAAQ,CAAC,EAAE,gBAAgB,CAAC;CAC7B;AAED,MAAM,WAAW,uBAAuB;IACtC;;;OAGG;IACH,aAAa,CAAC,EAAE,OAAO,CAAC;IACxB;;;OAGG;IACH,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB;;;;;OAKG;IACH,cAAc,CAAC,EAAE,SAAS,GAAG,OAAO,GAAG,UAAU,GAAG,OAAO,GAAG,QAAQ,CAAC;CACxE;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAC,iBAAiB,CAAC,CAAsC;WAEzD,0BAA0B,CAAC,QAAQ,EAAE,MAAM,iBAAiB,GAAG,SAAS,GAAG,IAAI;IAI7F,OAAO,CAAC,MAAM,CAAC,YAAY;IAO3B;;;;;;;;;;;OAWG;IAEH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,kBAAkB,CAoExC;IAKF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,kBAAkB,CAcxC;IAEF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,uBAAuB,CAkF7C;IAEF
|
|
1
|
+
{"version":3,"file":"contentValidator.d.ts","sourceRoot":"","sources":["../../src/security/contentValidator.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAOH,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,kCAAkC,CAAC;AAG1E,MAAM,MAAM,gBAAgB,GAAG,KAAK,GAAG,QAAQ,GAAG,MAAM,GAAG,UAAU,CAAC;AAEtE,MAAM,WAAW,uBAAuB;IACtC,OAAO,EAAE,OAAO,CAAC;IACjB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,gBAAgB,CAAC,EAAE,MAAM,EAAE,CAAC;IAC5B,QAAQ,CAAC,EAAE,gBAAgB,CAAC;CAC7B;AAED,MAAM,WAAW,uBAAuB;IACtC;;;OAGG;IACH,aAAa,CAAC,EAAE,OAAO,CAAC;IACxB;;;OAGG;IACH,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB;;;;;OAKG;IACH,cAAc,CAAC,EAAE,SAAS,GAAG,OAAO,GAAG,UAAU,GAAG,OAAO,GAAG,QAAQ,CAAC;CACxE;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAC,iBAAiB,CAAC,CAAsC;WAEzD,0BAA0B,CAAC,QAAQ,EAAE,MAAM,iBAAiB,GAAG,SAAS,GAAG,IAAI;IAI7F,OAAO,CAAC,MAAM,CAAC,YAAY;IAO3B;;;;;;;;;;;OAWG;IAEH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,kBAAkB,CAoExC;IAKF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,kBAAkB,CAcxC;IAEF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,uBAAuB,CAkF7C;IAEF;;;;;;;;OAQG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,oBAAoB,CAOzC;IAEH;;;;OAIG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,uBAAuB,CAK5C;IAEH;;;;;;;;;;;;;;;OAeG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,qBAAqB,CAc1C;IAEH;;;;OAIG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,qBAAqB,CAI1C;IAEH;;;;OAIG;IACH,OAAO,CAAC,MAAM,CAAC,uBAAuB;IAuCtC;;;;;;;;;OASG;IACH,OAAO,CAAC,MAAM,CAAC,sBAAsB;IAgErC;;;;;;;;OAQG;IACH,MAAM,CAAC,mBAAmB,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,GAAE,uBAA4B,GAAG,uBAAuB;IAiE3G;;;OAGG;IACH,MAAM,CAAC,mBAAmB,CAAC,WAAW,EAAE,MAAM,GAAG,OAAO;IA0JxD;;OAEG;IACH,MAAM,CAAC,gBAAgB,CAAC,QAAQ,EAAE,GAAG,GAAG,uBAAuB;IAuC/D;;OAEG;IACH,MAAM,CAAC,sBAAsB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM;CAkCvD"}
|
|
@@ -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,
|
|
199
|
-
*
|
|
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
|
|
273
|
-
|
|
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;
|
|
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 +
|
|
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
|
-
|
|
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"]}
|
package/dist/web/server.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../src/web/server.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAcH,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,mCAAmC,CAAC;AACvE,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,uCAAuC,CAAC;
|
|
1
|
+
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../src/web/server.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAcH,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,sCAAsC,CAAC;AAC1E,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,mCAAmC,CAAC;AACvE,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,uCAAuC,CAAC;AAa/E,qEAAqE;AACrE,wBAAgB,kBAAkB,IAAI,OAAO,CAE5C;AAED;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,sCAAsC;IACtC,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,qEAAqE;IACrE,YAAY,EAAE,MAAM,CAAC;IACrB,qEAAqE;IACrE,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB;;;;;OAKG;IACH,aAAa,CAAC,EAAE,aAAa,CAAC;IAC9B,kFAAkF;IAClF,UAAU,CAAC,EAAE,aAAa,CAAC;IAC3B,6FAA6F;IAC7F,WAAW,CAAC,EAAE,iBAAiB,CAAC;IAChC,gFAAgF;IAChF,iBAAiB,CAAC,EAAE,OAAO,SAAS,EAAE,MAAM,EAAE,CAAC;CAChD;AAED;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B,kFAAkF;IAClF,GAAG,CAAC,EAAE,OAAO,SAAS,EAAE,OAAO,CAAC;IAChC,2EAA2E;IAC3E,YAAY,CAAC,EAAE,CAAC,KAAK,EAAE,OAAO,qBAAqB,EAAE,eAAe,KAAK,IAAI,CAAC;IAC9E,iFAAiF;IACjF,iBAAiB,CAAC,EAAE,CAAC,QAAQ,EAAE,OAAO,qBAAqB,EAAE,cAAc,KAAK,IAAI,CAAC;CACtF;AAED;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,uCAAuC;IACvC,GAAG,EAAE,MAAM,CAAC;IACZ,4EAA4E;IAC5E,cAAc,EAAE,OAAO,CAAC;IACxB,kDAAkD;IAClD,aAAa,EAAE,OAAO,CAAC;IACvB,yDAAyD;IACzD,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAsCD;;;;;;;;;;;GAWG;AACH,wBAAsB,cAAc,CAAC,OAAO,EAAE,gBAAgB,GAAG,OAAO,CAAC,eAAe,CAAC,CAkLxF;AAED;;;;;;;;;;;;GAYG;AACH,wBAAsB,oBAAoB,CAAC,YAAY,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM,EAAE,aAAa,CAAC,EAAE,aAAa,EAAE,GAAG,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAuBvJ"}
|
package/dist/web/server.js
CHANGED
|
@@ -26,6 +26,8 @@ const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
|
26
26
|
const DEFAULT_PORT = 3939;
|
|
27
27
|
const CONSOLE_HOST = 'dollhouse.localhost';
|
|
28
28
|
const ALLOWED_PAGE_EXTENSIONS = new Set(['.html', '.htm']);
|
|
29
|
+
/** Max JSON body for setup routes (install/open-config). Ingest routes use their own 1mb limit. */
|
|
30
|
+
const SETUP_BODY_LIMIT = '1kb';
|
|
29
31
|
/** Track whether the web server is already running in-process. */
|
|
30
32
|
let serverRunning = false;
|
|
31
33
|
let serverPort = DEFAULT_PORT;
|
|
@@ -109,10 +111,11 @@ export async function startWebServer(options) {
|
|
|
109
111
|
next();
|
|
110
112
|
});
|
|
111
113
|
// Setup routes: auto-install DollhouseMCP to MCP clients (mount BEFORE API routes)
|
|
112
|
-
|
|
114
|
+
// Body limit scoped to setup routes only — ingest routes need 1mb for follower log forwarding
|
|
115
|
+
const setupJsonParser = express.json({ limit: SETUP_BODY_LIMIT, type: 'application/json' });
|
|
113
116
|
const { installHandler, openConfigHandler, versionHandler, mcpbRedirectHandler, detectHandler } = createSetupRoutes();
|
|
114
|
-
app.post('/api/setup/install', installHandler);
|
|
115
|
-
app.post('/api/setup/open-config', openConfigHandler);
|
|
117
|
+
app.post('/api/setup/install', setupJsonParser, installHandler);
|
|
118
|
+
app.post('/api/setup/open-config', setupJsonParser, openConfigHandler);
|
|
116
119
|
app.get('/api/setup/version', versionHandler);
|
|
117
120
|
app.get('/api/setup/mcpb', mcpbRedirectHandler);
|
|
118
121
|
app.get('/api/setup/detect', detectHandler);
|
|
@@ -196,11 +199,21 @@ export async function startWebServer(options) {
|
|
|
196
199
|
}
|
|
197
200
|
res.sendFile(join(publicDir, 'index.html'));
|
|
198
201
|
});
|
|
202
|
+
// Global error handler — catch Express errors and route to logger instead of terminal.
|
|
203
|
+
// Without this, Express dumps stack traces to stderr (visible in --web terminal).
|
|
204
|
+
// All errors still appear in the management console's Logs tab via MemoryLogSink.
|
|
205
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
206
|
+
app.use((err, _req, res, _next) => {
|
|
207
|
+
const status = err.status || err.statusCode || 500;
|
|
208
|
+
logger.warn(`[WebUI] ${err.name}: ${err.message}`);
|
|
209
|
+
if (!res.headersSent) {
|
|
210
|
+
res.status(status).json({ error: err.message });
|
|
211
|
+
}
|
|
212
|
+
});
|
|
199
213
|
// Bind to localhost only — handle port conflicts gracefully
|
|
200
|
-
// NOTE:
|
|
201
|
-
//
|
|
202
|
-
//
|
|
203
|
-
// for the human at the terminal, logger for the log viewer tab.
|
|
214
|
+
// NOTE: Use stderr for terminal output, not stdout. In MCP stdio mode, stdout
|
|
215
|
+
// is reserved for JSON-RPC messages — any non-JSON output corrupts the protocol.
|
|
216
|
+
// stderr is safe for human-readable messages in both MCP and standalone modes.
|
|
204
217
|
await new Promise((resolve) => {
|
|
205
218
|
const httpServer = app.listen(port, '127.0.0.1', () => {
|
|
206
219
|
serverRunning = true;
|
|
@@ -208,7 +221,7 @@ export async function startWebServer(options) {
|
|
|
208
221
|
const url = `http://${CONSOLE_HOST}:${port}`;
|
|
209
222
|
const fallbackUrl = `http://127.0.0.1:${port}`;
|
|
210
223
|
logger.info(`[WebUI] Management console running at ${url}`);
|
|
211
|
-
console.
|
|
224
|
+
console.error(`\n DollhouseMCP Management Console\n ${url}\n ${fallbackUrl} (fallback)\n`);
|
|
212
225
|
if (options.openBrowser) {
|
|
213
226
|
openInBrowser(url);
|
|
214
227
|
}
|
|
@@ -218,7 +231,7 @@ export async function startWebServer(options) {
|
|
|
218
231
|
if (err.code === 'EADDRINUSE') {
|
|
219
232
|
const url = `http://${CONSOLE_HOST}:${port}`;
|
|
220
233
|
logger.info(`[WebUI] Port ${port} already in use — opening existing console`);
|
|
221
|
-
console.
|
|
234
|
+
console.error(`\n DollhouseMCP Management Console (existing instance)\n ${url}\n`);
|
|
222
235
|
if (options.openBrowser) {
|
|
223
236
|
openInBrowser(url);
|
|
224
237
|
}
|
|
@@ -265,4 +278,4 @@ export async function openPortfolioBrowser(portfolioDir, port, mcpAqlHandler, ta
|
|
|
265
278
|
...(browserResult.error ? { warning: `Browser could not be opened automatically: ${browserResult.error}. Open ${url} manually.` } : {}),
|
|
266
279
|
};
|
|
267
280
|
}
|
|
268
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/web/server.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * DollhouseMCP Web UI Server\n *\n * Lightweight Express server for browsing portfolio elements in a browser.\n * Bound to 127.0.0.1 only (localhost). Read-only for V1.\n *\n * Can be started standalone (`--web` flag) or from within the MCP server\n * process via `openPortfolioBrowser()`.\n *\n * @see https://github.com/DollhouseMCP/mcp-server-v2-refactor/issues/704\n * @see https://github.com/DollhouseMCP/mcp-server-v2-refactor/issues/774\n */\n\nimport express from 'express';\nimport { join, dirname, extname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { execFile } from 'node:child_process';\nimport { platform } from 'node:os';\nimport { mkdir, readdir } from 'node:fs/promises';\nimport { createApiRoutes, createGatewayApiRoutes } from './routes.js';\nimport { createLogRoutes, type LogRoutesResult } from './routes/logRoutes.js';\nimport { createMetricsRoutes, type MetricsRoutesResult } from './routes/metricsRoutes.js';\nimport { createHealthRoutes } from './routes/healthRoutes.js';\nimport { createSetupRoutes } from './routes/setupRoutes.js';\nimport { logger } from '../utils/logger.js';\nimport type { MCPAQLHandler } from '../handlers/mcp-aql/MCPAQLHandler.js';\nimport type { MemoryLogSink } from '../logging/sinks/MemoryLogSink.js';\nimport type { MemoryMetricsSink } from '../metrics/sinks/MemoryMetricsSink.js';\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\nconst DEFAULT_PORT = 3939;\nconst CONSOLE_HOST = 'dollhouse.localhost';\nconst ALLOWED_PAGE_EXTENSIONS = new Set(['.html', '.htm']);\n\n/** Track whether the web server is already running in-process. */\nlet serverRunning = false;\nlet serverPort = DEFAULT_PORT;\n\n/** Check whether the web server has been started in this process. */\nexport function isWebServerRunning(): boolean {\n  return serverRunning;\n}\n\n/**\n * Options for starting the web server.\n */\nexport interface WebServerOptions {\n  /** Port to bind to (default: 3939) */\n  port?: number;\n  /** Path to the portfolio directory (e.g., ~/.dollhouse/portfolio) */\n  portfolioDir: string;\n  /** Open the browser automatically after starting (default: false) */\n  openBrowser?: boolean;\n  /**\n   * MCPAQLHandler for routing through the MCP-AQL pipeline.\n   * When provided, API routes use the gateway (validated, cached, gatekeeper-checked).\n   * When absent, falls back to direct filesystem access (legacy behavior).\n   * Issue #796: Web MCP-AQL Gateway.\n   */\n  mcpAqlHandler?: MCPAQLHandler;\n  /** MemoryLogSink for log routes (optional — logs tab disabled if not provided) */\n  memorySink?: MemoryLogSink;\n  /** MemoryMetricsSink for metrics routes (optional — metrics tab disabled if not provided) */\n  metricsSink?: MemoryMetricsSink;\n  /** Additional routers to mount before the SPA fallback (e.g., ingest routes) */\n  additionalRouters?: import('express').Router[];\n}\n\n/**\n * Result of starting the web server, including hooks for DI wiring.\n */\nexport interface WebServerResult {\n  /** Express app instance — for mounting additional routes (e.g., ingest routes) */\n  app?: import('express').Express;\n  /** Log broadcast function — call with each entry to push to SSE clients */\n  logBroadcast?: (entry: import('../logging/types.js').UnifiedLogEntry) => void;\n  /** Metrics snapshot function — call with each snapshot to push to SSE clients */\n  metricsOnSnapshot?: (snapshot: import('../metrics/types.js').MetricSnapshot) => void;\n}\n\n/**\n * Result of attempting to open the browser.\n */\nexport interface BrowserOpenResult {\n  /** The URL the server is running on */\n  url: string;\n  /** Whether the server was already running (true) or just started (false) */\n  alreadyRunning: boolean;\n  /** Whether the browser was successfully opened */\n  browserOpened: boolean;\n  /** Warning message if the browser could not be opened */\n  warning?: string;\n}\n\n/**\n * Open a URL in the system's default browser.\n *\n * Platform-aware:\n * - macOS: `open`\n * - Linux: `xdg-open`\n * - Windows: `start`\n *\n * @param url - The URL to open\n * @returns Promise that resolves to true if the browser opened, false with error message if not\n */\nfunction openInBrowser(url: string): Promise<{ success: boolean; error?: string }> {\n  return new Promise((resolve) => {\n    const plat = platform();\n    const cmd = plat === 'darwin' ? 'open'\n      : plat === 'win32' ? 'start'\n      : 'xdg-open';\n\n    // Security: use execFile with URL as argument array, not string interpolation\n    const urlStr = String(url);\n    // Accept localhost, 127.0.0.1, and *.localhost subdomains (RFC 6761)\n    if (!/^https?:\\/\\/(localhost|127\\.0\\.0\\.1|[\\w-]+\\.localhost)[:/]/.test(urlStr)) {\n      resolve({ success: false, error: 'URL must be a localhost HTTP URL' });\n      return;\n    }\n    execFile(cmd, [urlStr], (err) => {\n      if (err) {\n        logger.warn(`[WebUI] Could not auto-open browser: ${err.message}`);\n        resolve({ success: false, error: err.message });\n      } else {\n        resolve({ success: true });\n      }\n    });\n  });\n}\n\n/**\n * Start the portfolio web server.\n *\n * Binds to 127.0.0.1 only (localhost). Serves the portfolio browser\n * frontend and API routes for reading elements.\n *\n * Idempotent: if the server is already running, optionally opens the\n * browser without starting a second instance.\n *\n * @param options - Server configuration\n * @returns Hooks for DI wiring (log broadcast, metrics onSnapshot)\n */\nexport async function startWebServer(options: WebServerOptions): Promise<WebServerResult> {\n  const port = options.port || DEFAULT_PORT;\n  const result: WebServerResult = {};\n\n  if (serverRunning) {\n    if (options.openBrowser) {\n      openInBrowser(`http://${CONSOLE_HOST}:${serverPort}`);\n    }\n    return result;\n  }\n\n  const app = express();\n  result.app = app;\n  app.disable('x-powered-by');\n\n  // Security headers\n  app.use((_req, res, next) => {\n    res.setHeader('X-Content-Type-Options', 'nosniff');\n    res.setHeader('X-Frame-Options', 'DENY');\n    res.setHeader('X-XSS-Protection', '1; mode=block');\n    res.setHeader('Referrer-Policy', 'no-referrer');\n    res.setHeader('Access-Control-Allow-Origin', `http://${CONSOLE_HOST}:${port}`);\n    res.setHeader('Content-Security-Policy', [\n      \"default-src 'self'\",\n      \"script-src 'self' cdn.jsdelivr.net cdnjs.cloudflare.com\",\n      \"style-src 'self' 'unsafe-inline' cdnjs.cloudflare.com cdn.jsdelivr.net\",\n      \"connect-src 'self' raw.githubusercontent.com\",\n      \"font-src 'self'\",\n    ].join('; '));\n    next();\n  });\n\n  // Setup routes: auto-install DollhouseMCP to MCP clients (mount BEFORE API routes)\n  app.use(express.json({ limit: '1kb', type: 'application/json' }));\n  const { installHandler, openConfigHandler, versionHandler, mcpbRedirectHandler, detectHandler } = createSetupRoutes();\n  app.post('/api/setup/install', installHandler);\n  app.post('/api/setup/open-config', openConfigHandler);\n  app.get('/api/setup/version', versionHandler);\n  app.get('/api/setup/mcpb', mcpbRedirectHandler);\n  app.get('/api/setup/detect', detectHandler);\n  logger.info('[WebUI] Setup routes mounted at /api/setup');\n\n  // API routes — use MCP-AQL gateway when handler is available (Issue #796)\n  if (options.mcpAqlHandler) {\n    app.use('/api', createGatewayApiRoutes(options.mcpAqlHandler, options.portfolioDir));\n\n    // Permission evaluation routes (POST /evaluate_permission, GET /permissions/status)\n    const { registerPermissionRoutes } = await import('./routes/permissionRoutes.js');\n    const permRouter = (await import('express')).Router();\n    registerPermissionRoutes(permRouter, options.mcpAqlHandler);\n    app.use('/api', permRouter);\n\n    logger.info('[WebUI] API routes using MCP-AQL Gateway + permission routes');\n  } else {\n    app.use('/api', createApiRoutes(options.portfolioDir));\n    logger.warn('[WebUI] API routes using direct filesystem access (no MCP-AQL handler available)');\n  }\n\n  // Console routes: logs, metrics, health\n  let logRoutes: LogRoutesResult | undefined;\n  let metricsRoutes: MetricsRoutesResult | undefined;\n\n  if (options.memorySink) {\n    logRoutes = createLogRoutes(options.memorySink);\n    app.use('/api', logRoutes.router);\n    result.logBroadcast = logRoutes.broadcast;\n    logger.info('[WebUI] Log viewer routes mounted at /api/logs');\n  }\n\n  if (options.metricsSink) {\n    metricsRoutes = createMetricsRoutes(options.metricsSink);\n    app.use('/api', metricsRoutes.router);\n    result.metricsOnSnapshot = metricsRoutes.onSnapshot;\n    logger.info('[WebUI] Metrics routes mounted at /api/metrics');\n  }\n\n  if (options.memorySink) {\n    const healthRouter = createHealthRoutes({\n      memorySink: options.memorySink,\n      metricsSink: options.metricsSink,\n      logClientCount: logRoutes ? logRoutes.clientCount : () => 0,\n      metricsClientCount: metricsRoutes ? metricsRoutes.clientCount : () => 0,\n    });\n    app.use('/api', healthRouter);\n  }\n\n  // Serve ~/.dollhouse/pages/ at /pages/ — dashboards, generated content, stack views\n  const pagesDir = join(dirname(options.portfolioDir), 'pages');\n  mkdir(pagesDir, { recursive: true }).catch(err => {\n    logger.warn(`[WebUI] Could not create pages directory: ${(err as Error).message}`);\n  });\n  app.use('/pages', express.static(pagesDir));\n\n  /**\n   * GET /api/pages\n   * Lists available HTML pages in ~/.dollhouse/pages/.\n   * Returns page names and their URLs for the management console.\n   */\n  app.get('/api/pages', async (_req, res) => {\n    try {\n      const files = await readdir(pagesDir);\n      const pages = files\n        .filter(f => !f.startsWith('.') && ALLOWED_PAGE_EXTENSIONS.has(extname(f)))\n        .map(f => ({ name: f, url: `/pages/${f}` }));\n      res.json({ pages, directory: pagesDir });\n    } catch {\n      res.json({ pages: [], directory: pagesDir });\n    }\n  });\n\n  // Additional routers (e.g., unified console ingest routes) — must mount before SPA fallback\n  options.additionalRouters?.forEach(router => app.use(router));\n\n  // Static frontend files\n  const publicDir = join(__dirname, 'public');\n  app.use(express.static(publicDir));\n\n  // SPA fallback\n  app.get('/{*path}', (req, res) => {\n    const normalizedPath = req.path.normalize('NFC');\n    if (normalizedPath.startsWith('/api/')) {\n      res.status(404).json({ error: `API route not found: ${normalizedPath}` });\n      return;\n    }\n    if (normalizedPath.startsWith('/pages/')) {\n      res.status(404).json({ error: `Page not found: ${normalizedPath}` });\n      return;\n    }\n    res.sendFile(join(publicDir, 'index.html'));\n  });\n\n  // Bind to localhost only — handle port conflicts gracefully\n  // NOTE: console.log is intentional here (not logger). In --web standalone mode,\n  // the user sees terminal output directly. logger.info writes to the structured\n  // log system (MemoryLogSink → SSE → web console). Both are needed: console.log\n  // for the human at the terminal, logger for the log viewer tab.\n  await new Promise<void>((resolve) => {\n    const httpServer = app.listen(port, '127.0.0.1', () => {\n      serverRunning = true;\n      serverPort = port;\n      const url = `http://${CONSOLE_HOST}:${port}`;\n      const fallbackUrl = `http://127.0.0.1:${port}`;\n      logger.info(`[WebUI] Management console running at ${url}`);\n      console.log(`\\n  DollhouseMCP Management Console\\n  ${url}\\n  ${fallbackUrl} (fallback)\\n`);\n\n      if (options.openBrowser) {\n        openInBrowser(url);\n      }\n      resolve();\n    });\n    httpServer.on('error', (err: NodeJS.ErrnoException) => {\n      if (err.code === 'EADDRINUSE') {\n        const url = `http://${CONSOLE_HOST}:${port}`;\n        logger.info(`[WebUI] Port ${port} already in use — opening existing console`);\n        console.log(`\\n  DollhouseMCP Management Console (existing instance)\\n  ${url}\\n`);\n        if (options.openBrowser) {\n          openInBrowser(url);\n        }\n      } else {\n        logger.error(`[WebUI] Failed to bind port ${port}: ${err.message}`);\n      }\n      resolve(); // Web console is optional — don't block startup\n    });\n  });\n\n  return result;\n}\n\n/**\n * Open the portfolio browser from within the MCP server process.\n *\n * Starts the web server if not already running, then opens the system\n * browser to the portfolio UI. Returns a result object indicating\n * whether the server started and the browser opened successfully.\n *\n * Called by the `open_portfolio_browser` MCP-AQL operation (Issue #774).\n *\n * @param portfolioDir - Path to the portfolio directory (e.g., ~/.dollhouse/portfolio)\n * @param port - Port to bind to (default: 3939)\n * @returns Result with URL, server status, and browser open status\n */\nexport async function openPortfolioBrowser(portfolioDir: string, port?: number, mcpAqlHandler?: MCPAQLHandler, tab?: string): Promise<BrowserOpenResult> {\n  const targetPort = port || DEFAULT_PORT;\n  const baseUrl = `http://${CONSOLE_HOST}:${targetPort}`;\n  const url = tab ? `${baseUrl}/#${tab}` : baseUrl;\n  const alreadyRunning = serverRunning;\n\n  if (!serverRunning) {\n    await startWebServer({\n      portfolioDir,\n      port: targetPort,\n      openBrowser: false, // We'll open manually below to capture the result\n      mcpAqlHandler,\n    });\n  }\n\n  const browserResult = await openInBrowser(url);\n\n  return {\n    url,\n    alreadyRunning,\n    browserOpened: browserResult.success,\n    ...(browserResult.error ? { warning: `Browser could not be opened automatically: ${browserResult.error}. Open ${url} manually.` } : {}),\n  };\n}\n"]}
|
|
281
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"server.js","sourceRoot":"","sources":["../../src/web/server.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * DollhouseMCP Web UI Server\n *\n * Lightweight Express server for browsing portfolio elements in a browser.\n * Bound to 127.0.0.1 only (localhost). Read-only for V1.\n *\n * Can be started standalone (`--web` flag) or from within the MCP server\n * process via `openPortfolioBrowser()`.\n *\n * @see https://github.com/DollhouseMCP/mcp-server-v2-refactor/issues/704\n * @see https://github.com/DollhouseMCP/mcp-server-v2-refactor/issues/774\n */\n\nimport express from 'express';\nimport { join, dirname, extname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { execFile } from 'node:child_process';\nimport { platform } from 'node:os';\nimport { mkdir, readdir } from 'node:fs/promises';\nimport { createApiRoutes, createGatewayApiRoutes } from './routes.js';\nimport { createLogRoutes, type LogRoutesResult } from './routes/logRoutes.js';\nimport { createMetricsRoutes, type MetricsRoutesResult } from './routes/metricsRoutes.js';\nimport { createHealthRoutes } from './routes/healthRoutes.js';\nimport { createSetupRoutes } from './routes/setupRoutes.js';\nimport { logger } from '../utils/logger.js';\nimport type { MCPAQLHandler } from '../handlers/mcp-aql/MCPAQLHandler.js';\nimport type { MemoryLogSink } from '../logging/sinks/MemoryLogSink.js';\nimport type { MemoryMetricsSink } from '../metrics/sinks/MemoryMetricsSink.js';\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\nconst DEFAULT_PORT = 3939;\nconst CONSOLE_HOST = 'dollhouse.localhost';\nconst ALLOWED_PAGE_EXTENSIONS = new Set(['.html', '.htm']);\n/** Max JSON body for setup routes (install/open-config). Ingest routes use their own 1mb limit. */\nconst SETUP_BODY_LIMIT = '1kb';\n\n/** Track whether the web server is already running in-process. */\nlet serverRunning = false;\nlet serverPort = DEFAULT_PORT;\n\n/** Check whether the web server has been started in this process. */\nexport function isWebServerRunning(): boolean {\n  return serverRunning;\n}\n\n/**\n * Options for starting the web server.\n */\nexport interface WebServerOptions {\n  /** Port to bind to (default: 3939) */\n  port?: number;\n  /** Path to the portfolio directory (e.g., ~/.dollhouse/portfolio) */\n  portfolioDir: string;\n  /** Open the browser automatically after starting (default: false) */\n  openBrowser?: boolean;\n  /**\n   * MCPAQLHandler for routing through the MCP-AQL pipeline.\n   * When provided, API routes use the gateway (validated, cached, gatekeeper-checked).\n   * When absent, falls back to direct filesystem access (legacy behavior).\n   * Issue #796: Web MCP-AQL Gateway.\n   */\n  mcpAqlHandler?: MCPAQLHandler;\n  /** MemoryLogSink for log routes (optional — logs tab disabled if not provided) */\n  memorySink?: MemoryLogSink;\n  /** MemoryMetricsSink for metrics routes (optional — metrics tab disabled if not provided) */\n  metricsSink?: MemoryMetricsSink;\n  /** Additional routers to mount before the SPA fallback (e.g., ingest routes) */\n  additionalRouters?: import('express').Router[];\n}\n\n/**\n * Result of starting the web server, including hooks for DI wiring.\n */\nexport interface WebServerResult {\n  /** Express app instance — for mounting additional routes (e.g., ingest routes) */\n  app?: import('express').Express;\n  /** Log broadcast function — call with each entry to push to SSE clients */\n  logBroadcast?: (entry: import('../logging/types.js').UnifiedLogEntry) => void;\n  /** Metrics snapshot function — call with each snapshot to push to SSE clients */\n  metricsOnSnapshot?: (snapshot: import('../metrics/types.js').MetricSnapshot) => void;\n}\n\n/**\n * Result of attempting to open the browser.\n */\nexport interface BrowserOpenResult {\n  /** The URL the server is running on */\n  url: string;\n  /** Whether the server was already running (true) or just started (false) */\n  alreadyRunning: boolean;\n  /** Whether the browser was successfully opened */\n  browserOpened: boolean;\n  /** Warning message if the browser could not be opened */\n  warning?: string;\n}\n\n/**\n * Open a URL in the system's default browser.\n *\n * Platform-aware:\n * - macOS: `open`\n * - Linux: `xdg-open`\n * - Windows: `start`\n *\n * @param url - The URL to open\n * @returns Promise that resolves to true if the browser opened, false with error message if not\n */\nfunction openInBrowser(url: string): Promise<{ success: boolean; error?: string }> {\n  return new Promise((resolve) => {\n    const plat = platform();\n    const cmd = plat === 'darwin' ? 'open'\n      : plat === 'win32' ? 'start'\n      : 'xdg-open';\n\n    // Security: use execFile with URL as argument array, not string interpolation\n    const urlStr = String(url);\n    // Accept localhost, 127.0.0.1, and *.localhost subdomains (RFC 6761)\n    if (!/^https?:\\/\\/(localhost|127\\.0\\.0\\.1|[\\w-]+\\.localhost)[:/]/.test(urlStr)) {\n      resolve({ success: false, error: 'URL must be a localhost HTTP URL' });\n      return;\n    }\n    execFile(cmd, [urlStr], (err) => {\n      if (err) {\n        logger.warn(`[WebUI] Could not auto-open browser: ${err.message}`);\n        resolve({ success: false, error: err.message });\n      } else {\n        resolve({ success: true });\n      }\n    });\n  });\n}\n\n/**\n * Start the portfolio web server.\n *\n * Binds to 127.0.0.1 only (localhost). Serves the portfolio browser\n * frontend and API routes for reading elements.\n *\n * Idempotent: if the server is already running, optionally opens the\n * browser without starting a second instance.\n *\n * @param options - Server configuration\n * @returns Hooks for DI wiring (log broadcast, metrics onSnapshot)\n */\nexport async function startWebServer(options: WebServerOptions): Promise<WebServerResult> {\n  const port = options.port || DEFAULT_PORT;\n  const result: WebServerResult = {};\n\n  if (serverRunning) {\n    if (options.openBrowser) {\n      openInBrowser(`http://${CONSOLE_HOST}:${serverPort}`);\n    }\n    return result;\n  }\n\n  const app = express();\n  result.app = app;\n  app.disable('x-powered-by');\n\n  // Security headers\n  app.use((_req, res, next) => {\n    res.setHeader('X-Content-Type-Options', 'nosniff');\n    res.setHeader('X-Frame-Options', 'DENY');\n    res.setHeader('X-XSS-Protection', '1; mode=block');\n    res.setHeader('Referrer-Policy', 'no-referrer');\n    res.setHeader('Access-Control-Allow-Origin', `http://${CONSOLE_HOST}:${port}`);\n    res.setHeader('Content-Security-Policy', [\n      \"default-src 'self'\",\n      \"script-src 'self' cdn.jsdelivr.net cdnjs.cloudflare.com\",\n      \"style-src 'self' 'unsafe-inline' cdnjs.cloudflare.com cdn.jsdelivr.net\",\n      \"connect-src 'self' raw.githubusercontent.com\",\n      \"font-src 'self'\",\n    ].join('; '));\n    next();\n  });\n\n  // Setup routes: auto-install DollhouseMCP to MCP clients (mount BEFORE API routes)\n  // Body limit scoped to setup routes only — ingest routes need 1mb for follower log forwarding\n  const setupJsonParser = express.json({ limit: SETUP_BODY_LIMIT, type: 'application/json' });\n  const { installHandler, openConfigHandler, versionHandler, mcpbRedirectHandler, detectHandler } = createSetupRoutes();\n  app.post('/api/setup/install', setupJsonParser, installHandler);\n  app.post('/api/setup/open-config', setupJsonParser, openConfigHandler);\n  app.get('/api/setup/version', versionHandler);\n  app.get('/api/setup/mcpb', mcpbRedirectHandler);\n  app.get('/api/setup/detect', detectHandler);\n  logger.info('[WebUI] Setup routes mounted at /api/setup');\n\n  // API routes — use MCP-AQL gateway when handler is available (Issue #796)\n  if (options.mcpAqlHandler) {\n    app.use('/api', createGatewayApiRoutes(options.mcpAqlHandler, options.portfolioDir));\n\n    // Permission evaluation routes (POST /evaluate_permission, GET /permissions/status)\n    const { registerPermissionRoutes } = await import('./routes/permissionRoutes.js');\n    const permRouter = (await import('express')).Router();\n    registerPermissionRoutes(permRouter, options.mcpAqlHandler);\n    app.use('/api', permRouter);\n\n    logger.info('[WebUI] API routes using MCP-AQL Gateway + permission routes');\n  } else {\n    app.use('/api', createApiRoutes(options.portfolioDir));\n    logger.warn('[WebUI] API routes using direct filesystem access (no MCP-AQL handler available)');\n  }\n\n  // Console routes: logs, metrics, health\n  let logRoutes: LogRoutesResult | undefined;\n  let metricsRoutes: MetricsRoutesResult | undefined;\n\n  if (options.memorySink) {\n    logRoutes = createLogRoutes(options.memorySink);\n    app.use('/api', logRoutes.router);\n    result.logBroadcast = logRoutes.broadcast;\n    logger.info('[WebUI] Log viewer routes mounted at /api/logs');\n  }\n\n  if (options.metricsSink) {\n    metricsRoutes = createMetricsRoutes(options.metricsSink);\n    app.use('/api', metricsRoutes.router);\n    result.metricsOnSnapshot = metricsRoutes.onSnapshot;\n    logger.info('[WebUI] Metrics routes mounted at /api/metrics');\n  }\n\n  if (options.memorySink) {\n    const healthRouter = createHealthRoutes({\n      memorySink: options.memorySink,\n      metricsSink: options.metricsSink,\n      logClientCount: logRoutes ? logRoutes.clientCount : () => 0,\n      metricsClientCount: metricsRoutes ? metricsRoutes.clientCount : () => 0,\n    });\n    app.use('/api', healthRouter);\n  }\n\n  // Serve ~/.dollhouse/pages/ at /pages/ — dashboards, generated content, stack views\n  const pagesDir = join(dirname(options.portfolioDir), 'pages');\n  mkdir(pagesDir, { recursive: true }).catch(err => {\n    logger.warn(`[WebUI] Could not create pages directory: ${(err as Error).message}`);\n  });\n  app.use('/pages', express.static(pagesDir));\n\n  /**\n   * GET /api/pages\n   * Lists available HTML pages in ~/.dollhouse/pages/.\n   * Returns page names and their URLs for the management console.\n   */\n  app.get('/api/pages', async (_req, res) => {\n    try {\n      const files = await readdir(pagesDir);\n      const pages = files\n        .filter(f => !f.startsWith('.') && ALLOWED_PAGE_EXTENSIONS.has(extname(f)))\n        .map(f => ({ name: f, url: `/pages/${f}` }));\n      res.json({ pages, directory: pagesDir });\n    } catch {\n      res.json({ pages: [], directory: pagesDir });\n    }\n  });\n\n  // Additional routers (e.g., unified console ingest routes) — must mount before SPA fallback\n  options.additionalRouters?.forEach(router => app.use(router));\n\n  // Static frontend files\n  const publicDir = join(__dirname, 'public');\n  app.use(express.static(publicDir));\n\n  // SPA fallback\n  app.get('/{*path}', (req, res) => {\n    const normalizedPath = req.path.normalize('NFC');\n    if (normalizedPath.startsWith('/api/')) {\n      res.status(404).json({ error: `API route not found: ${normalizedPath}` });\n      return;\n    }\n    if (normalizedPath.startsWith('/pages/')) {\n      res.status(404).json({ error: `Page not found: ${normalizedPath}` });\n      return;\n    }\n    res.sendFile(join(publicDir, 'index.html'));\n  });\n\n  // Global error handler — catch Express errors and route to logger instead of terminal.\n  // Without this, Express dumps stack traces to stderr (visible in --web terminal).\n  // All errors still appear in the management console's Logs tab via MemoryLogSink.\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  app.use((err: Error, _req: import('express').Request, res: import('express').Response, _next: import('express').NextFunction) => {\n    const status = (err as any).status || (err as any).statusCode || 500;\n    logger.warn(`[WebUI] ${err.name}: ${err.message}`);\n    if (!res.headersSent) {\n      res.status(status).json({ error: err.message });\n    }\n  });\n\n  // Bind to localhost only — handle port conflicts gracefully\n  // NOTE: Use stderr for terminal output, not stdout. In MCP stdio mode, stdout\n  // is reserved for JSON-RPC messages — any non-JSON output corrupts the protocol.\n  // stderr is safe for human-readable messages in both MCP and standalone modes.\n  await new Promise<void>((resolve) => {\n    const httpServer = app.listen(port, '127.0.0.1', () => {\n      serverRunning = true;\n      serverPort = port;\n      const url = `http://${CONSOLE_HOST}:${port}`;\n      const fallbackUrl = `http://127.0.0.1:${port}`;\n      logger.info(`[WebUI] Management console running at ${url}`);\n      console.error(`\\n  DollhouseMCP Management Console\\n  ${url}\\n  ${fallbackUrl} (fallback)\\n`);\n\n      if (options.openBrowser) {\n        openInBrowser(url);\n      }\n      resolve();\n    });\n    httpServer.on('error', (err: NodeJS.ErrnoException) => {\n      if (err.code === 'EADDRINUSE') {\n        const url = `http://${CONSOLE_HOST}:${port}`;\n        logger.info(`[WebUI] Port ${port} already in use — opening existing console`);\n        console.error(`\\n  DollhouseMCP Management Console (existing instance)\\n  ${url}\\n`);\n        if (options.openBrowser) {\n          openInBrowser(url);\n        }\n      } else {\n        logger.error(`[WebUI] Failed to bind port ${port}: ${err.message}`);\n      }\n      resolve(); // Web console is optional — don't block startup\n    });\n  });\n\n  return result;\n}\n\n/**\n * Open the portfolio browser from within the MCP server process.\n *\n * Starts the web server if not already running, then opens the system\n * browser to the portfolio UI. Returns a result object indicating\n * whether the server started and the browser opened successfully.\n *\n * Called by the `open_portfolio_browser` MCP-AQL operation (Issue #774).\n *\n * @param portfolioDir - Path to the portfolio directory (e.g., ~/.dollhouse/portfolio)\n * @param port - Port to bind to (default: 3939)\n * @returns Result with URL, server status, and browser open status\n */\nexport async function openPortfolioBrowser(portfolioDir: string, port?: number, mcpAqlHandler?: MCPAQLHandler, tab?: string): Promise<BrowserOpenResult> {\n  const targetPort = port || DEFAULT_PORT;\n  const baseUrl = `http://${CONSOLE_HOST}:${targetPort}`;\n  const url = tab ? `${baseUrl}/#${tab}` : baseUrl;\n  const alreadyRunning = serverRunning;\n\n  if (!serverRunning) {\n    await startWebServer({\n      portfolioDir,\n      port: targetPort,\n      openBrowser: false, // We'll open manually below to capture the result\n      mcpAqlHandler,\n    });\n  }\n\n  const browserResult = await openInBrowser(url);\n\n  return {\n    url,\n    alreadyRunning,\n    browserOpened: browserResult.success,\n    ...(browserResult.error ? { warning: `Browser could not be opened automatically: ${browserResult.error}. Open ${url} manually.` } : {}),\n  };\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dollhousemcp/mcp-server",
|
|
3
|
-
"version": "2.0.
|
|
3
|
+
"version": "2.0.9",
|
|
4
4
|
"description": "DollhouseMCP - A Model Context Protocol (MCP) server that enables dynamic AI persona management from markdown files, allowing Claude and other compatible AI assistants to activate and switch between different behavioral personas.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "dist/index.js",
|
|
@@ -201,7 +201,7 @@
|
|
|
201
201
|
"packages/safety"
|
|
202
202
|
],
|
|
203
203
|
"engines": {
|
|
204
|
-
"node": ">=
|
|
204
|
+
"node": ">=18.0.0",
|
|
205
205
|
"npm": ">=10.0.0"
|
|
206
206
|
},
|
|
207
207
|
"overrides": {
|
package/server.json
CHANGED
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
"name": "io.github.DollhouseMCP/mcp-server",
|
|
4
4
|
"title": "DollhouseMCP",
|
|
5
5
|
"description": "OSS to create Personas, Skills, Templates, Agents, and Memories to customize your AI experience.",
|
|
6
|
-
"version": "2.0.
|
|
6
|
+
"version": "2.0.8",
|
|
7
7
|
"homepage": "https://dollhousemcp.com",
|
|
8
8
|
"repository": {
|
|
9
9
|
"type": "git",
|
|
@@ -29,7 +29,7 @@
|
|
|
29
29
|
{
|
|
30
30
|
"registryType": "npm",
|
|
31
31
|
"identifier": "@dollhousemcp/mcp-server",
|
|
32
|
-
"version": "2.0.
|
|
32
|
+
"version": "2.0.8",
|
|
33
33
|
"transport": {
|
|
34
34
|
"type": "stdio"
|
|
35
35
|
}
|