@dollhousemcp/mcp-server 2.0.12 → 2.0.13

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,17 @@
1
1
  # Changelog
2
2
 
3
+ ## [2.0.13] - 2026-04-13
4
+
5
+ ### Security Fixes
6
+
7
+ - **False-positive CRITICAL alerts eliminated** — bundled elements containing legitimate YAML keys (`javascript:`) or educational security payloads (`wget` in pentest templates) no longer fire CRITICAL alerts at install time (#1941)
8
+ - **JavaScript protocol injection regex tightened** — pattern now requires a non-whitespace character after the colon (`javascript[ \t]*:[ \t]*\S`), preventing bare YAML map keys from matching while still catching real `javascript:void(0)` injection attempts (#1941)
9
+ - **Bundled element trust system** — build generates `data/HASHES.json`; at startup, `DefaultElementProvider` verifies each bundled file against its SHA-256 and registers matching hashes with `ContentValidator`; verified content bypasses injection scanning, modified files automatically lose trust (#1941)
10
+
11
+ ### CI/CD
12
+
13
+ - **OIDC publishing restored** — removed broken `npm install -g npm@11` step that caused every publish workflow to fail with `MODULE_NOT_FOUND: promise-retry`; npm 10.x (ships with Node 22) already supports `--provenance` (#1941)
14
+
3
15
  ## [2.0.12] - 2026-04-12
4
16
 
5
17
  ### Authenticated Web Console
@@ -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.12";
6
- export declare const BUILD_TIMESTAMP = "2026-04-12T18:19:09.410Z";
5
+ export declare const PACKAGE_VERSION = "2.0.13";
6
+ export declare const BUILD_TIMESTAMP = "2026-04-13T16:33:05.030Z";
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.12';
6
- export const BUILD_TIMESTAMP = '2026-04-12T18:19:09.410Z';
5
+ export const PACKAGE_VERSION = '2.0.13';
6
+ export const BUILD_TIMESTAMP = '2026-04-13T16:33:05.030Z';
7
7
  export const BUILD_TYPE = 'npm';
8
8
  export const PACKAGE_NAME = '@dollhousemcp/mcp-server';
9
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsUUFBUSxDQUFDO0FBQ3hDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMTInO1xuZXhwb3J0IGNvbnN0IEJVSUxEX1RJTUVTVEFNUCA9ICcyMDI2LTA0LTEyVDE4OjE5OjA5LjQxMFonO1xuZXhwb3J0IGNvbnN0IEJVSUxEX1RZUEU6ICducG0nIHwgJ2dpdCcgPSAnbnBtJztcbmV4cG9ydCBjb25zdCBQQUNLQUdFX05BTUUgPSAnQGRvbGxob3VzZW1jcC9tY3Atc2VydmVyJztcbiJdfQ==
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsUUFBUSxDQUFDO0FBQ3hDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcyLjAuMTMnO1xuZXhwb3J0IGNvbnN0IEJVSUxEX1RJTUVTVEFNUCA9ICcyMDI2LTA0LTEzVDE2OjMzOjA1LjAzMFonO1xuZXhwb3J0IGNvbnN0IEJVSUxEX1RZUEU6ICducG0nIHwgJ2dpdCcgPSAnbnBtJztcbmV4cG9ydCBjb25zdCBQQUNLQUdFX05BTUUgPSAnQGRvbGxob3VzZW1jcC9tY3Atc2VydmVyJztcbiJdfQ==
@@ -37,6 +37,14 @@ export declare class DefaultElementProvider {
37
37
  private static metadataCache;
38
38
  private static readonly MAX_CACHE_SIZE;
39
39
  constructor(config?: DefaultElementProviderConfig);
40
+ /**
41
+ * Read data/HASHES.json (shipped with the npm package) and register each
42
+ * SHA-256 hash with ContentValidator as trusted bundled content.
43
+ *
44
+ * Only files whose on-disk hash matches the manifest entry are registered —
45
+ * any post-install modification breaks the hash and revokes trust automatically.
46
+ */
47
+ private registerBundledHashes;
40
48
  /**
41
49
  * Get the current loadTestData configuration value
42
50
  * @returns Whether test data loading is enabled
@@ -1 +1 @@
1
- {"version":3,"file":"DefaultElementProvider.d.ts","sourceRoot":"","sources":["../../src/portfolio/DefaultElementProvider.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"DefaultElementProvider.d.ts","sourceRoot":"","sources":["../../src/portfolio/DefaultElementProvider.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAcH,OAAO,EAAE,sBAAsB,EAAE,MAAM,sCAAsC,CAAC;AAI9E,eAAO,MAAM,cAAc;;;;;;;;;CASjB,CAAC;AAkBX,MAAM,WAAW,4BAA4B;IAC3C,2EAA2E;IAC3E,eAAe,CAAC,EAAE,MAAM,EAAE,CAAC;IAC3B,6DAA6D;IAC7D,eAAe,CAAC,EAAE,OAAO,CAAC;IAC1B,qFAAqF;IACrF,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,4DAA4D;IAC5D,cAAc,CAAC,EAAE,sBAAsB,CAAC;CACzC;AASD,qBAAa,sBAAsB;IACjC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAS;IACnC,OAAO,CAAC,MAAM,CAAC,aAAa,CAAuB;IACnD,OAAO,CAAC,MAAM,CAAC,kBAAkB,CAAyC;IAC1E,OAAO,CAAC,QAAQ,CAAC,MAAM,CAA+B;IACtD,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAC,CAAyB;IACzD,OAAO,CAAC,MAAM,CAAC,WAAW,CAAC,CAAwG;IAEnI,MAAM,CAAC,cAAc,CAAC,EAAE,EAAE,CAAC,KAAK,EAAE,OAAO,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,KAAK,IAAI,GAAG,MAAM,IAAI;IAM5I,OAAO,CAAC,MAAM,CAAC,aAAa,CAA8C;IAC1E,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,cAAc,CAAM;gBAEhC,MAAM,CAAC,EAAE,4BAA4B;IAuDjD;;;;;;OAMG;IACH,OAAO,CAAC,qBAAqB;IA+B7B;;;OAGG;IACH,IAAW,wBAAwB,IAAI,OAAO,CAE7C;IAED;;;OAGG;IACH,IAAW,iBAAiB,IAAI,OAAO,CAEtC;YAMa,QAAQ;YASR,OAAO;YAQP,UAAU;YAaV,aAAa;YAQb,UAAU;YAUV,SAAS;YAUT,YAAY;YAUZ,eAAe;IAY7B;;;OAGG;IACH,OAAO,KAAK,eAAe,GAwC1B;IAED;;;OAGG;YACW,iBAAiB;IAyC/B;;OAEG;YACW,eAAe;IAS7B;;;;;;;;OAQG;IACH,OAAO,CAAC,gBAAgB;IA8CxB;;;OAGG;IAGH;;;;OAIG;IA+BH;;;;;OAKG;IAMH;;;;;OAKG;IAEH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,UAAU,CAAgB;IAClD,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,aAAa,CAAM;IAC3C,OAAO,CAAC,MAAM,CAAC,eAAe,CAAsC;IAEpE,OAAO,CAAC,SAAS;IAejB,OAAO,CAAC,aAAa;IAYrB;;;;OAIG;WACW,OAAO,IAAI,IAAI;IAsB7B;;;;;OAKG;WACW,mBAAmB,IAAI;QACnC,UAAU,EAAE;YACV,IAAI,EAAE,MAAM,CAAC;YACb,MAAM,EAAE,MAAM,CAAC;YACf,OAAO,EAAE,MAAM,CAAC;YAChB,OAAO,EAAE,MAAM,CAAC;YAChB,QAAQ,EAAE,MAAM,CAAC;YACjB,WAAW,EAAE,MAAM,CAAC;SACrB,CAAC;QACF,aAAa,EAAE;YACb,IAAI,EAAE,MAAM,CAAC;YACb,OAAO,EAAE,MAAM,CAAC;SACjB,CAAC;KACH;IAqBD;;;;;;;;;;;OAWG;YACW,gBAAgB;IAgI9B;;;;;OAKG;YACW,yBAAyB;IAcvC;;;;OAIG;IACH,OAAO,CAAC,uBAAuB;IA2D/B;;;OAGG;YACW,gBAAgB;IAqJ9B;;;OAGG;IACH;;;;OAIG;YACW,iBAAiB;IA4B/B;;;;;OAKG;YACW,wBAAwB;IA8EtC;;;;OAIG;IACU,gBAAgB,CAAC,gBAAgB,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAsBtE;;;OAGG;YACW,iBAAiB;CAmGhC"}
@@ -17,6 +17,7 @@ import { ElementType } from './types.js';
17
17
  import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
18
18
  import { SecurityMonitor } from '../security/securityMonitor.js';
19
19
  import { SecureYamlParser } from '../security/secureYamlParser.js';
20
+ import { ContentValidator } from '../security/contentValidator.js';
20
21
  // File operation constants
21
22
  export const FILE_CONSTANTS = {
22
23
  ELEMENT_EXTENSION: '.md',
@@ -102,6 +103,47 @@ export class DefaultElementProvider {
102
103
  logger.info('[DefaultElementProvider] Development mode detected - test data loading disabled');
103
104
  logger.info('[DefaultElementProvider] To enable test data, set DOLLHOUSE_LOAD_TEST_DATA=true');
104
105
  }
106
+ // Register bundled element hashes with ContentValidator so verified bundled
107
+ // content does not trigger false-positive CRITICAL injection alerts at startup.
108
+ this.registerBundledHashes();
109
+ }
110
+ /**
111
+ * Read data/HASHES.json (shipped with the npm package) and register each
112
+ * SHA-256 hash with ContentValidator as trusted bundled content.
113
+ *
114
+ * Only files whose on-disk hash matches the manifest entry are registered —
115
+ * any post-install modification breaks the hash and revokes trust automatically.
116
+ */
117
+ registerBundledHashes() {
118
+ try {
119
+ const hashesPath = path.join(this.__dirname, '..', 'data', 'HASHES.json');
120
+ const raw = fsSync.readFileSync(hashesPath, 'utf-8');
121
+ const manifest = JSON.parse(raw);
122
+ const dataDir = path.join(this.__dirname, '..', 'data');
123
+ let registered = 0;
124
+ for (const [relPath, expectedHash] of Object.entries(manifest.files)) {
125
+ const fullPath = path.join(dataDir, relPath);
126
+ try {
127
+ const content = fsSync.readFileSync(fullPath);
128
+ const actualHash = createHash('sha256').update(content).digest('hex');
129
+ if (actualHash === expectedHash) {
130
+ ContentValidator.registerBundledHash(expectedHash);
131
+ registered++;
132
+ }
133
+ else {
134
+ logger.warn(`[DefaultElementProvider] Bundled file hash mismatch — treating as untrusted: ${relPath}`);
135
+ }
136
+ }
137
+ catch {
138
+ // File missing from this build's data/ — skip silently
139
+ }
140
+ }
141
+ logger.debug(`[DefaultElementProvider] Registered ${registered} verified bundled element hashes`);
142
+ }
143
+ catch {
144
+ // HASHES.json missing (older install, dev mode, or stripped package) — degrade gracefully
145
+ logger.debug('[DefaultElementProvider] data/HASHES.json not found — bundled element trust skipped');
146
+ }
105
147
  }
106
148
  /**
107
149
  * Get the current loadTestData configuration value
@@ -986,4 +1028,4 @@ export class DefaultElementProvider {
986
1028
  }
987
1029
  }
988
1030
  }
989
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DefaultElementProvider.js","sourceRoot":"","sources":["../../src/portfolio/DefaultElementProvider.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * DefaultElementProvider - Populates portfolio with default elements from bundled data\n * \n * This class handles copying default personas, skills, templates, and other elements\n * from the NPM package or Git repository to the user's portfolio on first run.\n * It ensures users have example content to work with immediately after installation.\n */\n\n// NOTE: fsSync is intentionally kept for synchronous development mode detection at module load time.\n// The IS_DEVELOPMENT_MODE check runs during module initialization before async operations are available.\n// All other file operations use FileOperationsService for centralized, secure file handling.\nimport * as fsSync from 'fs';\nimport * as path from 'path';\nimport { fileURLToPath } from 'url';\nimport { createHash } from 'crypto';\nimport { logger } from '../utils/logger.js';\nimport { ElementType } from './types.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\nimport { IFileOperationsService } from '../services/FileOperationsService.js';\n\n// File operation constants\nexport const FILE_CONSTANTS = {\n  ELEMENT_EXTENSION: '.md',\n  YAML_EXTENSION: '.yaml',\n  YML_EXTENSION: '.yml',\n  JSON_EXTENSION: '.json',\n  MAX_FILE_SIZE: 10 * 1024 * 1024, // 10MB max file size for safety\n  CHECKSUM_ALGORITHM: 'sha256',\n  FILE_PERMISSIONS: 0o644,\n  CHUNK_SIZE: 64 * 1024 // 64KB chunks for reading large files\n} as const;\n\n// Development mode detection\n// When running from a git clone, we don't want to auto-load test data\nconst IS_DEVELOPMENT_MODE = (() => {\n  try {\n    // Check if we're in a git repository (development mode)\n    const gitDir = path.join(process.cwd(), '.git');\n    return fsSync.existsSync(gitDir);\n  } catch {\n    return false;\n  }\n})();\n\n// Internal constants\nconst COPY_RETRY_ATTEMPTS = 3;\nconst COPY_RETRY_DELAY = 100; // ms\n\nexport interface DefaultElementProviderConfig {\n  /** Custom data directory paths to search (checked before default paths) */\n  customDataPaths?: string[];\n  /** Whether to use default search paths after custom paths */\n  useDefaultPaths?: boolean;\n  /** Whether to load test/example data from repository (default: false in dev mode) */\n  loadTestData?: boolean;\n  /** FileOperationsService for centralized file operations */\n  fileOperations?: IFileOperationsService;\n}\n\n// PERFORMANCE: Metadata cache with mtime-based invalidation\ninterface MetadataCacheEntry {\n  metadata: any;\n  mtime: number;\n  size: number;\n}\n\nexport class DefaultElementProvider {\n  private readonly __dirname: string;\n  private static cachedDataDir: string | null = null;\n  private static populateInProgress: Map<string, Promise<void>> = new Map();\n  private readonly config: DefaultElementProviderConfig;\n  private readonly fileOperations?: IFileOperationsService;\n  private static logListener?: (level: 'debug' | 'info' | 'warn' | 'error', message: string, data?: Record<string, unknown>) => void;\n\n  static addLogListener(fn: (level: 'debug' | 'info' | 'warn' | 'error', message: string, data?: Record<string, unknown>) => void): () => void {\n    this.logListener = fn;\n    return () => { this.logListener = undefined; };\n  }\n\n  // PERFORMANCE OPTIMIZATION: Cache metadata with file mtime for invalidation\n  private static metadataCache: Map<string, MetadataCacheEntry> = new Map();\n  private static readonly MAX_CACHE_SIZE = 20; // MEMORY LEAK FIX: Further reduced cache size to prevent accumulation during performance tests\n\n  constructor(config?: DefaultElementProviderConfig) {\n    const __filename = fileURLToPath(import.meta.url);\n    this.__dirname = path.dirname(__filename);\n\n    // Store fileOperations service if provided\n    this.fileOperations = config?.fileOperations;\n    \n    // Check environment variable for test data loading\n    const envLoadTestData = process.env.DOLLHOUSE_LOAD_TEST_DATA;\n    const loadTestDataFromEnv = envLoadTestData === 'true' || envLoadTestData === '1';\n    const disableTestDataFromEnv = envLoadTestData === 'false' || envLoadTestData === '0';\n    \n    // Check if we're in development mode (with respect to FORCE_PRODUCTION_MODE override)\n    const isDevMode = (() => {\n      // Respect FORCE_PRODUCTION_MODE override first\n      if (process.env.FORCE_PRODUCTION_MODE === 'true') {\n        return false; // Force production mode\n      }\n      if (process.env.FORCE_PRODUCTION_MODE === 'false') {\n        return true; // Force development mode\n      }\n      // Fall back to git detection\n      return IS_DEVELOPMENT_MODE;\n    })();\n    \n    // Determine loadTestData value\n    let computedLoadTestData: boolean;\n    if (loadTestDataFromEnv) {\n      // Environment explicitly enables test data\n      computedLoadTestData = true;\n    } else if (disableTestDataFromEnv) {\n      // Environment explicitly disables test data\n      computedLoadTestData = false;\n    } else {\n      // Default logic: enable in production, disable in development unless config overrides\n      computedLoadTestData = !isDevMode && (config?.loadTestData ?? true);\n    }\n    \n    this.config = {\n      useDefaultPaths: true,\n      ...config,\n      // Apply final loadTestData logic - environment variables and development mode take precedence\n      loadTestData: loadTestDataFromEnv ? true : disableTestDataFromEnv ? false : computedLoadTestData\n    };\n    \n    if (isDevMode && !this.config.loadTestData) {\n      logger.info('[DefaultElementProvider] Development mode detected - test data loading disabled');\n      logger.info('[DefaultElementProvider] To enable test data, set DOLLHOUSE_LOAD_TEST_DATA=true');\n    }\n  }\n  \n  /**\n   * Get the current loadTestData configuration value\n   * @returns Whether test data loading is enabled\n   */\n  public get isTestDataLoadingEnabled(): boolean {\n    return this.config.loadTestData ?? false;\n  }\n  \n  /**\n   * Get whether the system is in development mode\n   * @returns Whether running in development mode\n   */\n  public get isDevelopmentMode(): boolean {\n    return IS_DEVELOPMENT_MODE;\n  }\n\n  // ===== HELPER METHODS FOR FILE OPERATIONS =====\n  // These methods provide backward compatibility when fileOperations is not injected\n  // and centralize all file system calls through FileOperationsService when available\n\n  private async statFile(filePath: string): Promise<fsSync.Stats> {\n    if (this.fileOperations) {\n      return this.fileOperations.stat(filePath);\n    }\n    // Fallback: import fs/promises dynamically for backward compatibility\n    const { promises: fs } = await import('fs');\n    return fs.stat(filePath);\n  }\n\n  private async listDir(dirPath: string): Promise<string[]> {\n    if (this.fileOperations) {\n      return this.fileOperations.listDirectory(dirPath);\n    }\n    const { promises: fs } = await import('fs');\n    return fs.readdir(dirPath);\n  }\n\n  private async fileExists(filePath: string): Promise<boolean> {\n    if (this.fileOperations) {\n      return this.fileOperations.exists(filePath);\n    }\n    const { promises: fs } = await import('fs');\n    try {\n      await fs.access(filePath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  private async makeDirectory(dirPath: string): Promise<void> {\n    if (this.fileOperations) {\n      return this.fileOperations.createDirectory(dirPath);\n    }\n    const { promises: fs } = await import('fs');\n    await fs.mkdir(dirPath, { recursive: true });\n  }\n\n  private async copyFileOp(sourcePath: string, destPath: string): Promise<void> {\n    if (this.fileOperations) {\n      return this.fileOperations.copyFile(sourcePath, destPath, {\n        source: 'DefaultElementProvider.copyFileOp'\n      });\n    }\n    const { promises: fs } = await import('fs');\n    await fs.copyFile(sourcePath, destPath);\n  }\n\n  private async chmodFile(filePath: string, mode: number): Promise<void> {\n    if (this.fileOperations) {\n      return this.fileOperations.chmod(filePath, mode, {\n        source: 'DefaultElementProvider.chmodFile'\n      });\n    }\n    const { promises: fs } = await import('fs');\n    await fs.chmod(filePath, mode);\n  }\n\n  private async deleteFileOp(filePath: string): Promise<void> {\n    if (this.fileOperations) {\n      return this.fileOperations.deleteFile(filePath, undefined, {\n        source: 'DefaultElementProvider.deleteFileOp'\n      });\n    }\n    const { promises: fs } = await import('fs');\n    await fs.unlink(filePath);\n  }\n\n  private async readFileContent(filePath: string): Promise<string> {\n    if (this.fileOperations) {\n      return this.fileOperations.readFile(filePath, {\n        source: 'DefaultElementProvider.readFileContent'\n      });\n    }\n    const { promises: fs } = await import('fs');\n    return fs.readFile(filePath, 'utf-8');\n  }\n\n  // ===== END HELPER METHODS =====\n\n  /**\n   * Search paths for bundled data directory\n   * Ordered by priority - custom paths first, then development/git, then NPM locations\n   */\n  private get dataSearchPaths(): string[] {\n    const paths: string[] = [];\n    \n    // Add custom paths first (highest priority)\n    if (this.config.customDataPaths) {\n      paths.push(...this.config.customDataPaths);\n    }\n    \n    // Add default paths if enabled\n    if (this.config.useDefaultPaths !== false) {\n      // Skip development/repository data paths unless test data loading is enabled\n      if (this.config.loadTestData) {\n        // Development/Git installation (relative to this file)\n        paths.push(\n          path.join(this.__dirname, '../../data'),\n          path.join(this.__dirname, '../../../data'),\n          // Current working directory (last resort)\n          path.join(process.cwd(), 'data')\n        );\n      }\n      \n      // Always include NPM installation paths (these would have production data)\n      paths.push(\n        // NPM installations - macOS Homebrew\n        '/opt/homebrew/lib/node_modules/@dollhousemcp/mcp-server/data',\n        \n        // NPM installations - standard Unix/Linux\n        '/usr/local/lib/node_modules/@dollhousemcp/mcp-server/data',\n        '/usr/lib/node_modules/@dollhousemcp/mcp-server/data',\n        \n        // NPM installations - Windows\n        'C:\\\\Program Files\\\\nodejs\\\\node_modules\\\\@dollhousemcp\\\\mcp-server\\\\data',\n        'C:\\\\Program Files (x86)\\\\nodejs\\\\node_modules\\\\@dollhousemcp\\\\mcp-server\\\\data',\n        \n        // NPM installations - Windows with nvm\n        path.join(process.env.APPDATA || '', 'npm', 'node_modules', '@dollhousemcp', 'mcp-server', 'data')\n      );\n    }\n    \n    return paths;\n  }\n  \n  /**\n   * Find the bundled data directory by checking each search path\n   * Uses Promise.allSettled for better performance and caches the result\n   */\n  private async findDataDirectory(): Promise<string | null> {\n    // Return cached value if available\n    if (DefaultElementProvider.cachedDataDir !== null) {\n      return DefaultElementProvider.cachedDataDir;\n    }\n    \n    // Check all paths in parallel for better performance\n    const checkPromises = this.dataSearchPaths.map(async (searchPath) => {\n      try {\n        const stats = await this.statFile(searchPath);\n        if (stats.isDirectory()) {\n          // Verify it contains expected subdirectories\n          const hasPersonas = await this.directoryExists(path.join(searchPath, 'personas'));\n          const hasSkills = await this.directoryExists(path.join(searchPath, 'skills'));\n          if (hasPersonas || hasSkills) {\n            return searchPath;\n          }\n        }\n      } catch {\n        // Directory doesn't exist or can't be accessed\n        return null;\n      }\n      return null;\n    });\n    \n    const results = await Promise.allSettled(checkPromises);\n    \n    // Find the first successful result\n    for (const result of results) {\n      if (result.status === 'fulfilled' && result.value !== null) {\n        logger.info(`[DefaultElementProvider] Found data directory at: ${result.value}`);\n        // Cache the result\n        DefaultElementProvider.cachedDataDir = result.value;\n        return result.value;\n      }\n    }\n    \n    logger.warn('[DefaultElementProvider] No bundled data directory found in any search path');\n    return null;\n  }\n  \n  /**\n   * Helper to check if a directory exists\n   */\n  private async directoryExists(dirPath: string): Promise<boolean> {\n    try {\n      const stats = await this.statFile(dirPath);\n      return stats.isDirectory();\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Validate file path to prevent path traversal attacks\n   * SECURITY FIX: Added file path validation to prevent directory traversal\n   * Previously: File paths were used without validation, allowing potential ../../../ attacks\n   * Now: Strict validation ensures paths stay within allowed directories\n   * @param filePath The file path to validate\n   * @param allowedBasePaths Array of allowed base paths (optional)\n   * @returns true if path is safe, false otherwise\n   */\n  private validateFilePath(filePath: string, allowedBasePaths?: string[]): boolean {\n    try {\n      // SECURITY: Normalize path to prevent traversal attempts\n      const normalizedPath = path.normalize(filePath);\n      \n      // SECURITY: Reject paths containing traversal patterns\n      if (normalizedPath.includes('..')) {\n        logger.warn(`[DefaultElementProvider] Path traversal attempt blocked: ${filePath}`);\n        return false;\n      }\n      \n      // SECURITY: Check for home directory expansion attempts, but allow Windows 8.3 short path names\n      // Windows short path names use ~ followed by a digit (e.g., RUNNER~1), which should be allowed\n      // Only block ~ followed by / or \\ (home directory expansion patterns)\n      if (normalizedPath.includes('~/') || normalizedPath.includes('~\\\\')) {\n        logger.warn(`[DefaultElementProvider] Home directory expansion attempt blocked: ${filePath}`);\n        return false;\n      }\n      \n      // SECURITY: Reject absolute paths outside allowed directories\n      if (path.isAbsolute(normalizedPath) && allowedBasePaths) {\n        const isAllowed = allowedBasePaths.some(basePath => {\n          const normalizedBase = path.normalize(basePath);\n          return normalizedPath.startsWith(normalizedBase);\n        });\n        \n        if (!isAllowed) {\n          logger.warn(`[DefaultElementProvider] Absolute path outside allowed directories: ${filePath}`);\n          return false;\n        }\n      }\n      \n      // SECURITY: Reject null bytes and other dangerous characters\n      if (normalizedPath.includes('\\0') || normalizedPath.includes('\\x00')) { // NOSONAR - Null byte detection for security\n        logger.warn(`[DefaultElementProvider] Null byte in path blocked: ${filePath}`);\n        return false;\n      }\n      \n      return true;\n    } catch (error) {\n      logger.warn(`[DefaultElementProvider] Path validation error: ${error}`);\n      return false;\n    }\n  }\n\n  // DEPRECATED: Commented out filename pattern detection - replaced with metadata-based detection\n  /**\n   * Cached compiled regex patterns for performance optimization\n   * @deprecated Use metadata-based detection instead\n   */\n  // private static compiledTestPatterns: RegExp[] | null = null;\n\n  /**\n   * Get compiled test patterns with caching for better performance\n   * @deprecated Use metadata-based detection instead\n   * @returns Array of compiled regex patterns\n   */\n  // private getCompiledTestPatterns(): RegExp[] {\n  //   // Use cached patterns if available\n  //   if (DefaultElementProvider.compiledTestPatterns) {\n  //     return DefaultElementProvider.compiledTestPatterns;\n  //   }\n  //   \n  //   // Compile and cache patterns on first use\n  //   // CRITICAL FIX: Removed overly broad /^test-/i pattern that was blocking legitimate use\n  //   // Users should be able to create personas like \"test-driven-developer\" or \"test-automation-expert\"\n  //   // We only block specific test patterns that are clearly from our test suite\n  //   DefaultElementProvider.compiledTestPatterns = [\n  //     /^testpersona/i,              // Our test suite pattern\n  //     /^yamltest/i,                 // Security test pattern\n  //     /^yamlbomb/i,                 // Security test pattern\n  //     /^memory-test-/i,             // Performance test pattern\n  //     /^perf-test-/i,               // Performance test pattern\n  //     /^test-fixture-/i,            // Test fixture pattern (more specific)\n  //     /^test-data-/i,               // Test data pattern (more specific)\n  //     /bin-sh|rm-rf|pwned/i,        // Malicious patterns\n  //     /concurrent-\\d+/i,            // Concurrent test pattern\n  //     /legacy\\.md$/i,               // Legacy test pattern\n  //     /performance-test/i,          // Performance test pattern\n  //     /-\\d{13}-[a-z0-9]+\\.md$/i,    // Timestamp-based test files\n  //     /^unittest-/i,                // Unit test pattern\n  //     /^integrationtest-/i,         // Integration test pattern\n  //   ];\n  //   \n  //   return DefaultElementProvider.compiledTestPatterns;\n  // }\n\n  /**\n   * Check if a filename matches test data patterns that should never be copied to production\n   * @deprecated Use isDollhouseMCPTestElement() for metadata-based detection instead\n   * @param filename The filename to check\n   * @returns true if the filename matches test patterns that should be blocked\n   */\n  // private isTestDataPattern(filename: string): boolean {\n  //   const patterns = this.getCompiledTestPatterns();\n  //   return patterns.some(pattern => pattern.test(filename));\n  // }\n\n  /**\n   * Read metadata from YAML frontmatter only (never reads content body)\n   * Uses a small buffer to safely extract only the frontmatter between --- markers\n   * @param filePath Path to the file to read metadata from\n   * @returns Parsed metadata object or null if no frontmatter found\n   */\n  // PERFORMANCE OPTIMIZATION: Reusable buffer pool to reduce allocations\n  private static readonly bufferPool: Buffer[] = [];\n  private static readonly MAX_POOL_SIZE = 20; // MEMORY LEAK FIX: Reduced buffer pool size to match cache size for consistent memory management\n  private static bufferPoolStats = { hits: 0, misses: 0, created: 0 };\n  \n  private getBuffer(): Buffer {\n    // PERFORMANCE: Track buffer pool usage for optimization monitoring\n    let buffer = DefaultElementProvider.bufferPool.pop();\n    if (buffer) {\n      DefaultElementProvider.bufferPoolStats.hits++;\n      return buffer;\n    } else {\n      DefaultElementProvider.bufferPoolStats.misses++;\n      DefaultElementProvider.bufferPoolStats.created++;\n      buffer = Buffer.alloc(4096);\n      logger.debug(`[DefaultElementProvider] Created new buffer (pool empty), total created: ${DefaultElementProvider.bufferPoolStats.created}`);\n      return buffer;\n    }\n  }\n  \n  private releaseBuffer(buffer: Buffer): void {\n    // CRITICAL FIX: Always attempt to return buffer to pool for reuse\n    if (DefaultElementProvider.bufferPool.length < DefaultElementProvider.MAX_POOL_SIZE) {\n      buffer.fill(0); // SECURITY: Clear buffer before reuse to prevent data leakage\n      DefaultElementProvider.bufferPool.push(buffer);\n    } else {\n      // PERFORMANCE: If pool is full, clear the buffer to help GC\n      buffer.fill(0);\n      logger.debug('[DefaultElementProvider] Buffer pool full, discarding buffer');\n    }\n  }\n\n  /**\n   * Clean up buffer pool and cache to free memory\n   * PERFORMANCE FIX: Added cleanup method to prevent memory leaks\n   * This should be called during application shutdown or periodic cleanup\n   */\n  public static cleanup(): void {\n    // Clear buffer pool\n    DefaultElementProvider.bufferPool.length = 0;\n    \n    // Clear metadata cache\n    DefaultElementProvider.metadataCache.clear();\n    \n    // Clear cached data directory\n    DefaultElementProvider.cachedDataDir = null;\n    \n    // Clear population promises\n    DefaultElementProvider.populateInProgress.clear();\n    \n    logger.info('[DefaultElementProvider] Memory cleanup completed', {\n      bufferStats: DefaultElementProvider.bufferPoolStats,\n      cacheCleared: true\n    });\n    \n    // Reset stats\n    DefaultElementProvider.bufferPoolStats = { hits: 0, misses: 0, created: 0 };\n  }\n\n  /**\n   * Get performance statistics for monitoring\n   * PERFORMANCE MONITORING: Added statistics method for performance tracking\n   * This provides insights into buffer pool efficiency and cache performance\n   * @returns Object containing performance metrics\n   */\n  public static getPerformanceStats(): {\n    bufferPool: {\n      hits: number;\n      misses: number; \n      created: number;\n      hitRate: number;\n      poolSize: number;\n      maxPoolSize: number;\n    };\n    metadataCache: {\n      size: number;\n      maxSize: number;\n    };\n  } {\n    const bufferHits = DefaultElementProvider.bufferPoolStats.hits;\n    const bufferMisses = DefaultElementProvider.bufferPoolStats.misses;\n    const totalRequests = bufferHits + bufferMisses;\n\n    return {\n      bufferPool: {\n        hits: bufferHits,\n        misses: bufferMisses,\n        created: DefaultElementProvider.bufferPoolStats.created,\n        hitRate: totalRequests > 0 ? bufferHits / totalRequests : 0,\n        poolSize: DefaultElementProvider.bufferPool.length,\n        maxPoolSize: DefaultElementProvider.MAX_POOL_SIZE\n      },\n      metadataCache: {\n        size: DefaultElementProvider.metadataCache.size,\n        maxSize: DefaultElementProvider.MAX_CACHE_SIZE\n      }\n    };\n  }\n\n  /**\n   * Read only the YAML frontmatter metadata from a file (first 4KB)\n   *\n   * SECURITY NOTE (Issue #1228): This method MUST use validateContent: true when calling\n   * SecureYamlParser.parse() to ensure Unicode security validation runs. Setting validateContent\n   * to false would disable ALL content validation including zero-width character detection,\n   * creating a security bypass vulnerability.\n   *\n   * @param filePath - Absolute path to the file to read\n   * @param retries - Number of retries for transient failures (default: 2)\n   * @returns Parsed metadata object or null if no valid frontmatter found\n   */\n  private async readMetadataOnly(filePath: string, retries = 2): Promise<any | null> {\n    // PERFORMANCE: Check cache first before reading file\n    try {\n      const stats = await this.statFile(filePath);\n      const cacheKey = filePath;\n      const cached = DefaultElementProvider.metadataCache.get(cacheKey);\n\n      // Return cached metadata if file hasn't changed\n      // CRITICAL FIX: Use integer mtime comparison to avoid floating-point precision issues\n      if (cached && Math.floor(cached.mtime) === Math.floor(stats.mtimeMs) && cached.size === stats.size) {\n        logger.debug(`[DefaultElementProvider] Cache hit for ${filePath}`);\n        return cached.metadata;\n      }\n    } catch {\n      // File doesn't exist, proceed with normal flow\n    }\n    \n    try {\n      // NOTE: Using direct fs.open for partial file read (first 4KB only).\n      // FileOperationsService doesn't support partial reads with file handles.\n      // This is intentionally kept as a direct fs call for performance-critical\n      // metadata extraction that only needs the YAML frontmatter.\n      const { promises: fs } = await import('fs');\n      const fd = await fs.open(filePath, 'r');\n      // PERFORMANCE: Use buffer pool instead of allocating new buffer each time\n      const buffer = this.getBuffer();\n      \n      try {\n        const result = await fd.read(buffer, 0, 4096, 0);\n        const header = buffer.subarray(0, result.bytesRead).toString('utf-8');\n        \n        // Look for YAML frontmatter between --- markers\n        // Support both Unix (\\n) and Windows (\\r\\n) line endings\n        const match = header.match(/^---\\r?\\n([\\s\\S]*?)\\r?\\n---/);\n        if (!match) {\n          return null; // No frontmatter found\n        }\n        \n        // Parse the YAML frontmatter safely\n        try {\n          // SECURITY FIX (Issue #1228): Enable validateContent to block zero-width Unicode and other security threats\n          // SecureYamlParser provides additional validation, injection prevention, and content sanitization\n          // It expects full YAML with --- markers, so we reconstruct the frontmatter block\n          // We disable field-specific validation as this is general metadata parsing, not persona-specific\n          // but we MUST keep validateContent: true to ensure Unicode security validation runs\n          const fullYaml = `---\\n${match[1]}\\n---`;\n          const parseResult = SecureYamlParser.parse(fullYaml, {\n            // REQUIRED: validateContent must be true to enable Unicode security validation including\n            // zero-width character detection (U+200B-U+200F, U+FEFF, etc.). Setting this to false\n            // would disable ALL content validation, creating a security bypass for Unicode-based attacks.\n            // See UnicodeValidator.normalize() for detection logic.\n            validateContent: true,\n            validateFields: false   // Field-specific validation is optional for general metadata parsing\n          });\n          const metadata = parseResult.data;\n          \n          // PERFORMANCE: Cache the metadata with file stats for future reads\n          if (typeof metadata === 'object' && metadata !== null) {\n            try {\n              const stats = await this.statFile(filePath);\n              const cacheEntry: MetadataCacheEntry = {\n                metadata,\n                mtime: stats.mtimeMs,\n                size: stats.size\n              };\n              \n              // CRITICAL MEMORY LEAK FIX: More aggressive cache management to prevent unbounded growth\n              // Check if this entry already exists and just update it instead of adding new\n              if (DefaultElementProvider.metadataCache.has(filePath)) {\n                // Update existing entry - no eviction needed\n                DefaultElementProvider.metadataCache.set(filePath, cacheEntry);\n                logger.debug(`[DefaultElementProvider] Updated existing cache entry for ${filePath}`);\n              } else {\n                // New entry - check if we need to evict first\n                // Use > instead of >= to ensure we never exceed MAX_CACHE_SIZE\n                if (DefaultElementProvider.metadataCache.size >= DefaultElementProvider.MAX_CACHE_SIZE) {\n                  // More aggressive eviction: remove enough entries to stay well under limit\n                  const entriesToEvict = Math.max(1, Math.floor(DefaultElementProvider.MAX_CACHE_SIZE * 0.4));\n                  const keysToEvict = Array.from(DefaultElementProvider.metadataCache.keys()).slice(0, entriesToEvict);\n                  for (const key of keysToEvict) {\n                    DefaultElementProvider.metadataCache.delete(key);\n                  }\n                  logger.debug(`[DefaultElementProvider] Evicted ${keysToEvict.length} cache entries to manage memory (cache size was ${DefaultElementProvider.metadataCache.size + keysToEvict.length})`);\n                }\n                \n                DefaultElementProvider.metadataCache.set(filePath, cacheEntry);\n                logger.debug(`[DefaultElementProvider] Added new cache entry for ${filePath} (cache size now: ${DefaultElementProvider.metadataCache.size})`);\n              }\n            } catch {\n              // Ignore cache errors, return metadata anyway\n            }\n            return metadata;\n          }\n          return null;\n        } catch (yamlError) {\n          // Invalid YAML, return null\n          // ENHANCEMENT: Include error type for better debugging\n          const yamlErrorType = (yamlError as any)?.constructor?.name || 'YAMLError';\n          logger.debug(`[DefaultElementProvider] Invalid YAML in ${filePath}: ${yamlErrorType} - ${yamlError}`);\n          return null;\n        }\n      } finally {\n        // CRITICAL FIX: Ensure file descriptor is closed and buffer is released in ALL paths\n        try {\n          await fd.close();\n        } catch (closeError) {\n          logger.debug(`[DefaultElementProvider] Error closing file descriptor for ${filePath}: ${closeError}`);\n        }\n        // PERFORMANCE: Return buffer to pool for reuse\n        this.releaseBuffer(buffer);\n      }\n    } catch (error: any) {\n      // ENHANCEMENT: Include error type in debug logs for better debugging\n      const errorType = error?.constructor?.name || 'UnknownError';\n      const errorCode = error?.code || 'NO_CODE';\n      \n      // RELIABILITY: Add retry logic for transient failures\n      if (retries > 0 && (errorCode === 'EBUSY' || errorCode === 'EAGAIN')) {\n        logger.debug(`[DefaultElementProvider] Retrying read for ${filePath} after ${errorType}:${errorCode}`);\n        await new Promise(resolve => setTimeout(resolve, 50)); // Brief delay before retry\n        return this.readMetadataOnly(filePath, retries - 1);\n      }\n      \n      logger.debug(`[DefaultElementProvider] Could not read metadata from ${filePath}: ${errorType}:${errorCode} - ${error?.message || error}`);\n      return null;\n    }\n  }\n\n  /**\n   * Check if a file is a DollhouseMCP test element based on metadata\n   * This replaces filename pattern detection with accurate metadata-based detection\n   * @param filePath Path to the file to check\n   * @returns true if the file contains _dollhouseMCPTest: true metadata\n   */\n  private async isDollhouseMCPTestElement(filePath: string): Promise<boolean> {\n    try {\n      const metadata = await this.readMetadataOnly(filePath);\n      const isTest = !!(metadata && metadata._dollhouseMCPTest === true);\n      \n      \n      return isTest;\n    } catch (error) {\n      // If we can't read the metadata, assume it's not a test file\n      logger.debug(`[DefaultElementProvider] Error checking test metadata for ${filePath}: ${error}`);\n      return false;\n    }\n  }\n\n  /**\n   * Detect if we're in a production environment by checking for production indicators\n   * Uses a confidence-based approach requiring multiple indicators for better accuracy\n   * @returns true if this appears to be a production environment\n   */\n  private isProductionEnvironment(): boolean {\n    // Allow tests to explicitly override production mode detection\n    if (process.env.FORCE_PRODUCTION_MODE === 'true') {\n      return true;\n    }\n    if (process.env.FORCE_PRODUCTION_MODE === 'false') {\n      return false;\n    }\n    \n    // Weighted indicators for production detection\n    const indicators = {\n      // Strong indicators (weight: 2)\n      hasUserHomeDir: (process.env.HOME && (process.env.HOME.includes('/Users/') || process.env.HOME.includes('/home/'))) || \n                      !!process.env.USERPROFILE,\n      isProductionNode: process.env.NODE_ENV === 'production',\n      notInTestDir: (() => {\n        const cwd = process.cwd().toLowerCase();\n        // Normalize path separators for cross-platform checking (Windows uses \\ but checks use /)\n        const normalizedCwd = cwd.replaceAll('\\\\', '/');\n        return !normalizedCwd.includes('/test') && \n               !normalizedCwd.includes('/__tests__') && \n               !normalizedCwd.includes('/temp') &&\n               !normalizedCwd.includes('/dist/test');\n      })(),\n      \n      // Moderate indicators (weight: 1)\n      notInCI: !process.env.CI,\n      noTestEnv: process.env.NODE_ENV !== 'test',\n      noDevEnv: process.env.NODE_ENV !== 'development',\n    };\n    \n    // Calculate weighted score\n    let score = 0;\n    if (indicators.hasUserHomeDir) score += 2;\n    if (indicators.isProductionNode) score += 2;\n    if (indicators.notInTestDir) score += 2;\n    if (indicators.notInCI) score += 1;\n    if (indicators.noTestEnv) score += 1;\n    if (indicators.noDevEnv) score += 1;\n    \n    // Log detection details for debugging\n    const activeIndicators = Object.entries(indicators)\n      .filter(([_, value]) => value)\n      .map(([key]) => key);\n    \n    // TYPESCRIPT FIX: Removed logger.isDebugEnabled() check as this method doesn't exist on MCPLogger\n    // The logger already handles debug level internally, so we can call debug() directly\n    if (score >= 3) {\n      logger.debug(\n        '[DefaultElementProvider] Production environment detected',\n        { score, activeIndicators, forceMode: 'not set' }\n      );\n    }\n    \n    // Require a score of at least 3 for production detection (more confident)\n    // This prevents false positives in edge cases while maintaining security\n    return score >= 3;\n  }\n  \n  /**\n   * Copy all files from source directory to destination directory\n   * Skips files that already exist to preserve user modifications\n   */\n  private async copyElementFiles(sourceDir: string, destDir: string, elementType: string): Promise<number> {\n    let copiedCount = 0;\n    \n    \n    try {\n      // Ensure destination directory exists\n      await this.makeDirectory(destDir);\n\n      // Read source directory\n      const files = await this.listDir(sourceDir);\n      \n      \n      for (const file of files) {\n        \n        // Only copy markdown files\n        if (!file.endsWith(FILE_CONSTANTS.ELEMENT_EXTENSION)) {\n          continue;\n        }\n        \n        // Normalize filename for security\n        const normalizedFile = UnicodeValidator.normalize(file);\n        if (!normalizedFile.isValid) {\n          logger.warn(`[DefaultElementProvider] Skipping file with invalid Unicode: ${file}`);\n          continue;\n        }\n\n        const sourcePath = path.join(sourceDir, normalizedFile.normalizedContent);\n        const destPath = path.join(destDir, normalizedFile.normalizedContent);\n        \n        \n        // SECURITY FIX: Validate file paths to prevent path traversal attacks\n        // This prevents malicious files from escaping the intended directory structure\n        const sourceValid = this.validateFilePath(sourcePath, [sourceDir]);\n        const destValid = this.validateFilePath(destPath, [destDir]);\n        \n        if (!sourceValid || !destValid) {\n          logger.warn(\n            `[DefaultElementProvider] Skipping file with invalid path: ${normalizedFile.normalizedContent}`,\n            { sourcePath, destPath, elementType }\n          );\n          continue;\n        }\n        \n        // Production safety check: Block DollhouseMCP test elements in production environments\n        // Skip this check if loadTestData is explicitly enabled (for testing scenarios)\n        if (!this.config.loadTestData && this.isProductionEnvironment()) {\n          const isDollhouseTest = await this.isDollhouseMCPTestElement(sourcePath);\n          \n          if (isDollhouseTest) {\n            logger.warn(\n              `[DefaultElementProvider] SECURITY: Blocking DollhouseMCP test element in production: ${normalizedFile.normalizedContent}`,\n              { \n                file: normalizedFile.normalizedContent,\n                reason: 'DollhouseMCP test element detected in production environment',\n                elementType\n              }\n            );\n            \n            // Log security event for blocked test data\n            SecurityMonitor.logSecurityEvent({\n              type: 'TEST_DATA_BLOCKED',\n              severity: 'MEDIUM',\n              source: 'DefaultElementProvider.copyElementFiles',\n              details: `Blocked DollhouseMCP test element in production: ${normalizedFile.normalizedContent}`,\n              metadata: {\n                filename: normalizedFile.normalizedContent,\n                elementType,\n                reason: 'DollhouseMCP test element detected in production environment',\n                detectionMethod: 'metadata-based'\n              }\n            });\n            \n            continue;\n          }\n        }\n        \n        // Check if destination file already exists\n        const destExists = await this.fileExists(destPath);\n        if (destExists) {\n          logger.debug(`[DefaultElementProvider] Skipping existing file: ${normalizedFile.normalizedContent}`);\n          continue;\n        }\n        \n        try {\n          // Validate source file before copying\n          const sourceStats = await this.statFile(sourcePath);\n\n          // Check file size limit\n          if (sourceStats.size > FILE_CONSTANTS.MAX_FILE_SIZE) {\n            logger.warn(\n              `[DefaultElementProvider] Skipping oversized file ${normalizedFile.normalizedContent}: ` +\n              `${sourceStats.size} bytes (max: ${FILE_CONSTANTS.MAX_FILE_SIZE} bytes)`,\n              { \n                file: normalizedFile.normalizedContent, \n                size: sourceStats.size,\n                maxSize: FILE_CONSTANTS.MAX_FILE_SIZE,\n                elementType \n              }\n            );\n            continue;\n          }\n          \n          // Copy the file with verification\n          \n          await this.copyFileWithVerification(sourcePath, destPath);\n          copiedCount++;\n          \n          logger.debug(`[DefaultElementProvider] Copied ${elementType}: ${normalizedFile.normalizedContent}`);\n          \n          // Log security event for each file copied\n          SecurityMonitor.logSecurityEvent({\n            type: 'FILE_COPIED',\n            severity: 'LOW',\n            source: 'DefaultElementProvider.copyElementFiles',\n            details: `Copied default ${elementType} file: ${normalizedFile.normalizedContent}`,\n            metadata: {\n              sourcePath,\n              destPath,\n              elementType,\n              fileSize: (await this.statFile(destPath)).size\n            }\n          });\n        } catch (error) {\n          const err = error as Error;\n          logger.error(\n            `[DefaultElementProvider] Failed to copy ${normalizedFile.normalizedContent}`,\n            { \n              error: err.message,\n              stack: err.stack,\n              sourcePath,\n              destPath,\n              elementType\n            }\n          );\n          // Continue with other files instead of failing completely\n        }\n      }\n      \n      if (copiedCount > 0) {\n        logger.info(`[DefaultElementProvider] Copied ${copiedCount} ${elementType} file(s)`);\n      }\n      \n    } catch (error) {\n      logger.error(`[DefaultElementProvider] Error copying ${elementType} files:`, error);\n    }\n    \n    return copiedCount;\n  }\n  \n  /**\n   * Copy a file with integrity verification\n   * Ensures the file was copied correctly by comparing sizes\n   */\n  /**\n   * Calculate checksum of a file for integrity verification\n   * @param filePath Path to the file\n   * @returns Hex-encoded checksum\n   */\n  private async calculateChecksum(filePath: string): Promise<string> {\n    const hash = createHash(FILE_CONSTANTS.CHECKSUM_ALGORITHM);\n    // NOTE: Using direct fs.open for chunked/streaming file read.\n    // FileOperationsService doesn't support file handles for chunked reading.\n    // This is intentionally kept as a direct fs call for memory-efficient\n    // checksum calculation on potentially large files.\n    const { promises: fs } = await import('fs');\n    const stream = await fs.open(filePath, 'r');\n\n    try {\n      const buffer = Buffer.alloc(FILE_CONSTANTS.CHUNK_SIZE);\n      let bytesRead: number;\n\n      do {\n        const result = await stream.read(buffer, 0, FILE_CONSTANTS.CHUNK_SIZE);\n        bytesRead = result.bytesRead;\n\n        if (bytesRead > 0) {\n          hash.update(buffer.subarray(0, bytesRead));\n        }\n      } while (bytesRead > 0);\n\n      return hash.digest('hex');\n    } finally {\n      await stream.close();\n    }\n  }\n\n  /**\n   * Copy file with integrity verification and retry logic\n   * @param sourcePath Source file path\n   * @param destPath Destination file path\n   * @throws Error if copy fails after all retry attempts\n   */\n  private async copyFileWithVerification(sourcePath: string, destPath: string): Promise<void> {\n    let lastError: Error | null = null;\n\n    for (let attempt = 1; attempt <= COPY_RETRY_ATTEMPTS; attempt++) {\n      try {\n        // Copy the file\n        await this.copyFileOp(sourcePath, destPath);\n\n        // Verify size matches\n        const [sourceStats, destStats] = await Promise.all([\n          this.statFile(sourcePath),\n          this.statFile(destPath)\n        ]);\n\n        if (sourceStats.size !== destStats.size) {\n          throw new Error(\n            `Size mismatch after copy - source: ${sourceStats.size} bytes, ` +\n            `destination: ${destStats.size} bytes`\n          );\n        }\n\n        // Verify checksum matches for complete integrity\n        const [sourceChecksum, destChecksum] = await Promise.all([\n          this.calculateChecksum(sourcePath),\n          this.calculateChecksum(destPath)\n        ]);\n\n        if (sourceChecksum !== destChecksum) {\n          throw new Error(\n            `Checksum mismatch after copy - source: ${sourceChecksum}, ` +\n            `destination: ${destChecksum}`\n          );\n        }\n\n        // Set proper permissions\n        try {\n          await this.chmodFile(destPath, FILE_CONSTANTS.FILE_PERMISSIONS);\n        } catch (error) {\n          logger.debug(\n            `[DefaultElementProvider] Could not set permissions on ${destPath}: ${error}`,\n            { sourcePath, destPath, attempt }\n          );\n        }\n\n        // Success - file copied and verified\n        logger.debug(\n          `[DefaultElementProvider] Successfully copied and verified: ${path.basename(sourcePath)}`,\n          { size: sourceStats.size, checksum: sourceChecksum.substring(0, 8) }\n        );\n        return;\n\n      } catch (error) {\n        lastError = error as Error;\n\n        // Clean up failed copy\n        try {\n          await this.deleteFileOp(destPath);\n        } catch {\n          // Ignore cleanup errors\n        }\n\n        if (attempt < COPY_RETRY_ATTEMPTS) {\n          logger.debug(\n            `[DefaultElementProvider] Copy attempt ${attempt} failed, retrying...`,\n            { error: lastError.message, sourcePath, destPath }\n          );\n          await new Promise(resolve => setTimeout(resolve, COPY_RETRY_DELAY * attempt));\n        }\n      }\n    }\n    \n    // All attempts failed\n    throw new Error(\n      `Failed to copy ${path.basename(sourcePath)} after ${COPY_RETRY_ATTEMPTS} attempts: ` +\n      `${lastError?.message || 'Unknown error'}`\n    );\n  }\n  \n  /**\n   * Populate the portfolio with default elements from bundled data\n   * This is called during portfolio initialization for new installations\n   * Protected against concurrent calls to prevent race conditions\n   */\n  public async populateDefaults(portfolioBaseDir: string): Promise<void> {\n    // Check if population is already in progress for this portfolio\n    const existingPopulation = DefaultElementProvider.populateInProgress.get(portfolioBaseDir);\n    if (existingPopulation) {\n      logger.debug(\n        '[DefaultElementProvider] Population already in progress for portfolio, waiting...',\n        { portfolioBaseDir }\n      );\n      return existingPopulation;\n    }\n    \n    // Create new population promise\n    const populationPromise = this.performPopulation(portfolioBaseDir)\n      .finally(() => {\n        // Clean up when done\n        DefaultElementProvider.populateInProgress.delete(portfolioBaseDir);\n      });\n    \n    DefaultElementProvider.populateInProgress.set(portfolioBaseDir, populationPromise);\n    return populationPromise;\n  }\n  \n  /**\n   * Perform the actual population of default elements\n   * @param portfolioBaseDir Base directory of the portfolio\n   */\n  private async performPopulation(portfolioBaseDir: string): Promise<void> {\n    // Check if test data loading is disabled\n    // Note: This check is needed even though constructor sets config, because\n    // config can be overridden after construction\n    \n    // Use production environment detection that respects FORCE_PRODUCTION_MODE\n    const isDevelopmentMode = !this.isProductionEnvironment();\n    \n    if (isDevelopmentMode && !this.config.loadTestData) {\n      logger.info(\n        '[DefaultElementProvider] Skipping default element population in development mode',\n        { \n          portfolioBaseDir,\n          reason: 'Test data loading disabled',\n          enableWith: 'Set DOLLHOUSE_LOAD_TEST_DATA=true to enable'\n        }\n      );\n      return;\n    }\n    \n    logger.info(\n      '[DefaultElementProvider] Starting default element population',\n      { portfolioBaseDir }\n    );\n    \n    // Log security event for portfolio initialization\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_INITIALIZATION',\n      severity: 'LOW',\n      source: 'DefaultElementProvider.performPopulation',\n      details: `Starting default element population for portfolio: ${portfolioBaseDir}`\n    });\n    \n    // Find the bundled data directory\n    const dataDir = await this.findDataDirectory();\n    if (!dataDir) {\n      logger.warn(\n        '[DefaultElementProvider] No bundled data directory found - portfolio will start empty',\n        { \n          searchPaths: this.dataSearchPaths.slice(0, 3), // Log first few paths for debugging\n          cwd: process.cwd(),\n          dirname: this.__dirname\n        }\n      );\n      return;\n    }\n    \n    // Track total files copied\n    let totalCopied = 0;\n    const copiedCounts: Record<string, number> = {};\n    \n    // Copy each element type - directories now match enum values (all plural)\n    for (const elementType of Object.values(ElementType)) {\n      const sourceDir = path.join(dataDir, elementType);\n      const destDir = path.join(portfolioBaseDir, elementType);\n\n      // Check if source directory exists\n      const sourceDirExists = await this.fileExists(sourceDir);\n      if (!sourceDirExists) {\n        // Source directory doesn't exist, skip\n        logger.debug(`[DefaultElementProvider] No ${elementType} directory in bundled data`);\n        continue;\n      }\n\n      try {\n        const copiedCount = await this.copyElementFiles(sourceDir, destDir, elementType);\n        copiedCounts[elementType] = copiedCount;\n        totalCopied += copiedCount;\n      } catch (error) {\n        logger.error(`[DefaultElementProvider] Error copying ${elementType} elements:`, error);\n      }\n    }\n    \n    if (totalCopied > 0) {\n      logger.info(\n        `[DefaultElementProvider] Successfully populated portfolio with ${totalCopied} default element(s)`,\n        {\n          portfolioBaseDir,\n          dataDir,\n          breakdown: copiedCounts\n        }\n      );\n      \n      // Log security event for successful population\n      SecurityMonitor.logSecurityEvent({\n        type: 'PORTFOLIO_POPULATED',\n        severity: 'LOW',\n        source: 'DefaultElementProvider.performPopulation',\n        details: `Successfully populated portfolio with ${totalCopied} default elements`,\n        metadata: {\n          portfolioBaseDir,\n          dataDir,\n          copiedCounts\n        }\n      });\n    } else {\n      logger.info('[DefaultElementProvider] No new elements to copy - portfolio may already have content');\n    }\n  }\n}"]}
1031
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DefaultElementProvider.js","sourceRoot":"","sources":["../../src/portfolio/DefaultElementProvider.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * DefaultElementProvider - Populates portfolio with default elements from bundled data\n * \n * This class handles copying default personas, skills, templates, and other elements\n * from the NPM package or Git repository to the user's portfolio on first run.\n * It ensures users have example content to work with immediately after installation.\n */\n\n// NOTE: fsSync is intentionally kept for synchronous development mode detection at module load time.\n// The IS_DEVELOPMENT_MODE check runs during module initialization before async operations are available.\n// All other file operations use FileOperationsService for centralized, secure file handling.\nimport * as fsSync from 'fs';\nimport * as path from 'path';\nimport { fileURLToPath } from 'url';\nimport { createHash } from 'crypto';\nimport { logger } from '../utils/logger.js';\nimport { ElementType } from './types.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\nimport { IFileOperationsService } from '../services/FileOperationsService.js';\nimport { ContentValidator } from '../security/contentValidator.js';\n\n// File operation constants\nexport const FILE_CONSTANTS = {\n  ELEMENT_EXTENSION: '.md',\n  YAML_EXTENSION: '.yaml',\n  YML_EXTENSION: '.yml',\n  JSON_EXTENSION: '.json',\n  MAX_FILE_SIZE: 10 * 1024 * 1024, // 10MB max file size for safety\n  CHECKSUM_ALGORITHM: 'sha256',\n  FILE_PERMISSIONS: 0o644,\n  CHUNK_SIZE: 64 * 1024 // 64KB chunks for reading large files\n} as const;\n\n// Development mode detection\n// When running from a git clone, we don't want to auto-load test data\nconst IS_DEVELOPMENT_MODE = (() => {\n  try {\n    // Check if we're in a git repository (development mode)\n    const gitDir = path.join(process.cwd(), '.git');\n    return fsSync.existsSync(gitDir);\n  } catch {\n    return false;\n  }\n})();\n\n// Internal constants\nconst COPY_RETRY_ATTEMPTS = 3;\nconst COPY_RETRY_DELAY = 100; // ms\n\nexport interface DefaultElementProviderConfig {\n  /** Custom data directory paths to search (checked before default paths) */\n  customDataPaths?: string[];\n  /** Whether to use default search paths after custom paths */\n  useDefaultPaths?: boolean;\n  /** Whether to load test/example data from repository (default: false in dev mode) */\n  loadTestData?: boolean;\n  /** FileOperationsService for centralized file operations */\n  fileOperations?: IFileOperationsService;\n}\n\n// PERFORMANCE: Metadata cache with mtime-based invalidation\ninterface MetadataCacheEntry {\n  metadata: any;\n  mtime: number;\n  size: number;\n}\n\nexport class DefaultElementProvider {\n  private readonly __dirname: string;\n  private static cachedDataDir: string | null = null;\n  private static populateInProgress: Map<string, Promise<void>> = new Map();\n  private readonly config: DefaultElementProviderConfig;\n  private readonly fileOperations?: IFileOperationsService;\n  private static logListener?: (level: 'debug' | 'info' | 'warn' | 'error', message: string, data?: Record<string, unknown>) => void;\n\n  static addLogListener(fn: (level: 'debug' | 'info' | 'warn' | 'error', message: string, data?: Record<string, unknown>) => void): () => void {\n    this.logListener = fn;\n    return () => { this.logListener = undefined; };\n  }\n\n  // PERFORMANCE OPTIMIZATION: Cache metadata with file mtime for invalidation\n  private static metadataCache: Map<string, MetadataCacheEntry> = new Map();\n  private static readonly MAX_CACHE_SIZE = 20; // MEMORY LEAK FIX: Further reduced cache size to prevent accumulation during performance tests\n\n  constructor(config?: DefaultElementProviderConfig) {\n    const __filename = fileURLToPath(import.meta.url);\n    this.__dirname = path.dirname(__filename);\n\n    // Store fileOperations service if provided\n    this.fileOperations = config?.fileOperations;\n    \n    // Check environment variable for test data loading\n    const envLoadTestData = process.env.DOLLHOUSE_LOAD_TEST_DATA;\n    const loadTestDataFromEnv = envLoadTestData === 'true' || envLoadTestData === '1';\n    const disableTestDataFromEnv = envLoadTestData === 'false' || envLoadTestData === '0';\n    \n    // Check if we're in development mode (with respect to FORCE_PRODUCTION_MODE override)\n    const isDevMode = (() => {\n      // Respect FORCE_PRODUCTION_MODE override first\n      if (process.env.FORCE_PRODUCTION_MODE === 'true') {\n        return false; // Force production mode\n      }\n      if (process.env.FORCE_PRODUCTION_MODE === 'false') {\n        return true; // Force development mode\n      }\n      // Fall back to git detection\n      return IS_DEVELOPMENT_MODE;\n    })();\n    \n    // Determine loadTestData value\n    let computedLoadTestData: boolean;\n    if (loadTestDataFromEnv) {\n      // Environment explicitly enables test data\n      computedLoadTestData = true;\n    } else if (disableTestDataFromEnv) {\n      // Environment explicitly disables test data\n      computedLoadTestData = false;\n    } else {\n      // Default logic: enable in production, disable in development unless config overrides\n      computedLoadTestData = !isDevMode && (config?.loadTestData ?? true);\n    }\n    \n    this.config = {\n      useDefaultPaths: true,\n      ...config,\n      // Apply final loadTestData logic - environment variables and development mode take precedence\n      loadTestData: loadTestDataFromEnv ? true : disableTestDataFromEnv ? false : computedLoadTestData\n    };\n    \n    if (isDevMode && !this.config.loadTestData) {\n      logger.info('[DefaultElementProvider] Development mode detected - test data loading disabled');\n      logger.info('[DefaultElementProvider] To enable test data, set DOLLHOUSE_LOAD_TEST_DATA=true');\n    }\n\n    // Register bundled element hashes with ContentValidator so verified bundled\n    // content does not trigger false-positive CRITICAL injection alerts at startup.\n    this.registerBundledHashes();\n  }\n\n  /**\n   * Read data/HASHES.json (shipped with the npm package) and register each\n   * SHA-256 hash with ContentValidator as trusted bundled content.\n   *\n   * Only files whose on-disk hash matches the manifest entry are registered —\n   * any post-install modification breaks the hash and revokes trust automatically.\n   */\n  private registerBundledHashes(): void {\n    try {\n      const hashesPath = path.join(this.__dirname, '..', 'data', 'HASHES.json');\n      const raw = fsSync.readFileSync(hashesPath, 'utf-8');\n      const manifest: { files: Record<string, string> } = JSON.parse(raw);\n      const dataDir = path.join(this.__dirname, '..', 'data');\n      let registered = 0;\n\n      for (const [relPath, expectedHash] of Object.entries(manifest.files)) {\n        const fullPath = path.join(dataDir, relPath);\n        try {\n          const content = fsSync.readFileSync(fullPath);\n          const actualHash = createHash('sha256').update(content).digest('hex');\n          if (actualHash === expectedHash) {\n            ContentValidator.registerBundledHash(expectedHash);\n            registered++;\n          } else {\n            logger.warn(`[DefaultElementProvider] Bundled file hash mismatch — treating as untrusted: ${relPath}`);\n          }\n        } catch {\n          // File missing from this build's data/ — skip silently\n        }\n      }\n\n      logger.debug(`[DefaultElementProvider] Registered ${registered} verified bundled element hashes`);\n    } catch {\n      // HASHES.json missing (older install, dev mode, or stripped package) — degrade gracefully\n      logger.debug('[DefaultElementProvider] data/HASHES.json not found — bundled element trust skipped');\n    }\n  }\n  \n  /**\n   * Get the current loadTestData configuration value\n   * @returns Whether test data loading is enabled\n   */\n  public get isTestDataLoadingEnabled(): boolean {\n    return this.config.loadTestData ?? false;\n  }\n  \n  /**\n   * Get whether the system is in development mode\n   * @returns Whether running in development mode\n   */\n  public get isDevelopmentMode(): boolean {\n    return IS_DEVELOPMENT_MODE;\n  }\n\n  // ===== HELPER METHODS FOR FILE OPERATIONS =====\n  // These methods provide backward compatibility when fileOperations is not injected\n  // and centralize all file system calls through FileOperationsService when available\n\n  private async statFile(filePath: string): Promise<fsSync.Stats> {\n    if (this.fileOperations) {\n      return this.fileOperations.stat(filePath);\n    }\n    // Fallback: import fs/promises dynamically for backward compatibility\n    const { promises: fs } = await import('fs');\n    return fs.stat(filePath);\n  }\n\n  private async listDir(dirPath: string): Promise<string[]> {\n    if (this.fileOperations) {\n      return this.fileOperations.listDirectory(dirPath);\n    }\n    const { promises: fs } = await import('fs');\n    return fs.readdir(dirPath);\n  }\n\n  private async fileExists(filePath: string): Promise<boolean> {\n    if (this.fileOperations) {\n      return this.fileOperations.exists(filePath);\n    }\n    const { promises: fs } = await import('fs');\n    try {\n      await fs.access(filePath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  private async makeDirectory(dirPath: string): Promise<void> {\n    if (this.fileOperations) {\n      return this.fileOperations.createDirectory(dirPath);\n    }\n    const { promises: fs } = await import('fs');\n    await fs.mkdir(dirPath, { recursive: true });\n  }\n\n  private async copyFileOp(sourcePath: string, destPath: string): Promise<void> {\n    if (this.fileOperations) {\n      return this.fileOperations.copyFile(sourcePath, destPath, {\n        source: 'DefaultElementProvider.copyFileOp'\n      });\n    }\n    const { promises: fs } = await import('fs');\n    await fs.copyFile(sourcePath, destPath);\n  }\n\n  private async chmodFile(filePath: string, mode: number): Promise<void> {\n    if (this.fileOperations) {\n      return this.fileOperations.chmod(filePath, mode, {\n        source: 'DefaultElementProvider.chmodFile'\n      });\n    }\n    const { promises: fs } = await import('fs');\n    await fs.chmod(filePath, mode);\n  }\n\n  private async deleteFileOp(filePath: string): Promise<void> {\n    if (this.fileOperations) {\n      return this.fileOperations.deleteFile(filePath, undefined, {\n        source: 'DefaultElementProvider.deleteFileOp'\n      });\n    }\n    const { promises: fs } = await import('fs');\n    await fs.unlink(filePath);\n  }\n\n  private async readFileContent(filePath: string): Promise<string> {\n    if (this.fileOperations) {\n      return this.fileOperations.readFile(filePath, {\n        source: 'DefaultElementProvider.readFileContent'\n      });\n    }\n    const { promises: fs } = await import('fs');\n    return fs.readFile(filePath, 'utf-8');\n  }\n\n  // ===== END HELPER METHODS =====\n\n  /**\n   * Search paths for bundled data directory\n   * Ordered by priority - custom paths first, then development/git, then NPM locations\n   */\n  private get dataSearchPaths(): string[] {\n    const paths: string[] = [];\n    \n    // Add custom paths first (highest priority)\n    if (this.config.customDataPaths) {\n      paths.push(...this.config.customDataPaths);\n    }\n    \n    // Add default paths if enabled\n    if (this.config.useDefaultPaths !== false) {\n      // Skip development/repository data paths unless test data loading is enabled\n      if (this.config.loadTestData) {\n        // Development/Git installation (relative to this file)\n        paths.push(\n          path.join(this.__dirname, '../../data'),\n          path.join(this.__dirname, '../../../data'),\n          // Current working directory (last resort)\n          path.join(process.cwd(), 'data')\n        );\n      }\n      \n      // Always include NPM installation paths (these would have production data)\n      paths.push(\n        // NPM installations - macOS Homebrew\n        '/opt/homebrew/lib/node_modules/@dollhousemcp/mcp-server/data',\n        \n        // NPM installations - standard Unix/Linux\n        '/usr/local/lib/node_modules/@dollhousemcp/mcp-server/data',\n        '/usr/lib/node_modules/@dollhousemcp/mcp-server/data',\n        \n        // NPM installations - Windows\n        'C:\\\\Program Files\\\\nodejs\\\\node_modules\\\\@dollhousemcp\\\\mcp-server\\\\data',\n        'C:\\\\Program Files (x86)\\\\nodejs\\\\node_modules\\\\@dollhousemcp\\\\mcp-server\\\\data',\n        \n        // NPM installations - Windows with nvm\n        path.join(process.env.APPDATA || '', 'npm', 'node_modules', '@dollhousemcp', 'mcp-server', 'data')\n      );\n    }\n    \n    return paths;\n  }\n  \n  /**\n   * Find the bundled data directory by checking each search path\n   * Uses Promise.allSettled for better performance and caches the result\n   */\n  private async findDataDirectory(): Promise<string | null> {\n    // Return cached value if available\n    if (DefaultElementProvider.cachedDataDir !== null) {\n      return DefaultElementProvider.cachedDataDir;\n    }\n    \n    // Check all paths in parallel for better performance\n    const checkPromises = this.dataSearchPaths.map(async (searchPath) => {\n      try {\n        const stats = await this.statFile(searchPath);\n        if (stats.isDirectory()) {\n          // Verify it contains expected subdirectories\n          const hasPersonas = await this.directoryExists(path.join(searchPath, 'personas'));\n          const hasSkills = await this.directoryExists(path.join(searchPath, 'skills'));\n          if (hasPersonas || hasSkills) {\n            return searchPath;\n          }\n        }\n      } catch {\n        // Directory doesn't exist or can't be accessed\n        return null;\n      }\n      return null;\n    });\n    \n    const results = await Promise.allSettled(checkPromises);\n    \n    // Find the first successful result\n    for (const result of results) {\n      if (result.status === 'fulfilled' && result.value !== null) {\n        logger.info(`[DefaultElementProvider] Found data directory at: ${result.value}`);\n        // Cache the result\n        DefaultElementProvider.cachedDataDir = result.value;\n        return result.value;\n      }\n    }\n    \n    logger.warn('[DefaultElementProvider] No bundled data directory found in any search path');\n    return null;\n  }\n  \n  /**\n   * Helper to check if a directory exists\n   */\n  private async directoryExists(dirPath: string): Promise<boolean> {\n    try {\n      const stats = await this.statFile(dirPath);\n      return stats.isDirectory();\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Validate file path to prevent path traversal attacks\n   * SECURITY FIX: Added file path validation to prevent directory traversal\n   * Previously: File paths were used without validation, allowing potential ../../../ attacks\n   * Now: Strict validation ensures paths stay within allowed directories\n   * @param filePath The file path to validate\n   * @param allowedBasePaths Array of allowed base paths (optional)\n   * @returns true if path is safe, false otherwise\n   */\n  private validateFilePath(filePath: string, allowedBasePaths?: string[]): boolean {\n    try {\n      // SECURITY: Normalize path to prevent traversal attempts\n      const normalizedPath = path.normalize(filePath);\n      \n      // SECURITY: Reject paths containing traversal patterns\n      if (normalizedPath.includes('..')) {\n        logger.warn(`[DefaultElementProvider] Path traversal attempt blocked: ${filePath}`);\n        return false;\n      }\n      \n      // SECURITY: Check for home directory expansion attempts, but allow Windows 8.3 short path names\n      // Windows short path names use ~ followed by a digit (e.g., RUNNER~1), which should be allowed\n      // Only block ~ followed by / or \\ (home directory expansion patterns)\n      if (normalizedPath.includes('~/') || normalizedPath.includes('~\\\\')) {\n        logger.warn(`[DefaultElementProvider] Home directory expansion attempt blocked: ${filePath}`);\n        return false;\n      }\n      \n      // SECURITY: Reject absolute paths outside allowed directories\n      if (path.isAbsolute(normalizedPath) && allowedBasePaths) {\n        const isAllowed = allowedBasePaths.some(basePath => {\n          const normalizedBase = path.normalize(basePath);\n          return normalizedPath.startsWith(normalizedBase);\n        });\n        \n        if (!isAllowed) {\n          logger.warn(`[DefaultElementProvider] Absolute path outside allowed directories: ${filePath}`);\n          return false;\n        }\n      }\n      \n      // SECURITY: Reject null bytes and other dangerous characters\n      if (normalizedPath.includes('\\0') || normalizedPath.includes('\\x00')) { // NOSONAR - Null byte detection for security\n        logger.warn(`[DefaultElementProvider] Null byte in path blocked: ${filePath}`);\n        return false;\n      }\n      \n      return true;\n    } catch (error) {\n      logger.warn(`[DefaultElementProvider] Path validation error: ${error}`);\n      return false;\n    }\n  }\n\n  // DEPRECATED: Commented out filename pattern detection - replaced with metadata-based detection\n  /**\n   * Cached compiled regex patterns for performance optimization\n   * @deprecated Use metadata-based detection instead\n   */\n  // private static compiledTestPatterns: RegExp[] | null = null;\n\n  /**\n   * Get compiled test patterns with caching for better performance\n   * @deprecated Use metadata-based detection instead\n   * @returns Array of compiled regex patterns\n   */\n  // private getCompiledTestPatterns(): RegExp[] {\n  //   // Use cached patterns if available\n  //   if (DefaultElementProvider.compiledTestPatterns) {\n  //     return DefaultElementProvider.compiledTestPatterns;\n  //   }\n  //   \n  //   // Compile and cache patterns on first use\n  //   // CRITICAL FIX: Removed overly broad /^test-/i pattern that was blocking legitimate use\n  //   // Users should be able to create personas like \"test-driven-developer\" or \"test-automation-expert\"\n  //   // We only block specific test patterns that are clearly from our test suite\n  //   DefaultElementProvider.compiledTestPatterns = [\n  //     /^testpersona/i,              // Our test suite pattern\n  //     /^yamltest/i,                 // Security test pattern\n  //     /^yamlbomb/i,                 // Security test pattern\n  //     /^memory-test-/i,             // Performance test pattern\n  //     /^perf-test-/i,               // Performance test pattern\n  //     /^test-fixture-/i,            // Test fixture pattern (more specific)\n  //     /^test-data-/i,               // Test data pattern (more specific)\n  //     /bin-sh|rm-rf|pwned/i,        // Malicious patterns\n  //     /concurrent-\\d+/i,            // Concurrent test pattern\n  //     /legacy\\.md$/i,               // Legacy test pattern\n  //     /performance-test/i,          // Performance test pattern\n  //     /-\\d{13}-[a-z0-9]+\\.md$/i,    // Timestamp-based test files\n  //     /^unittest-/i,                // Unit test pattern\n  //     /^integrationtest-/i,         // Integration test pattern\n  //   ];\n  //   \n  //   return DefaultElementProvider.compiledTestPatterns;\n  // }\n\n  /**\n   * Check if a filename matches test data patterns that should never be copied to production\n   * @deprecated Use isDollhouseMCPTestElement() for metadata-based detection instead\n   * @param filename The filename to check\n   * @returns true if the filename matches test patterns that should be blocked\n   */\n  // private isTestDataPattern(filename: string): boolean {\n  //   const patterns = this.getCompiledTestPatterns();\n  //   return patterns.some(pattern => pattern.test(filename));\n  // }\n\n  /**\n   * Read metadata from YAML frontmatter only (never reads content body)\n   * Uses a small buffer to safely extract only the frontmatter between --- markers\n   * @param filePath Path to the file to read metadata from\n   * @returns Parsed metadata object or null if no frontmatter found\n   */\n  // PERFORMANCE OPTIMIZATION: Reusable buffer pool to reduce allocations\n  private static readonly bufferPool: Buffer[] = [];\n  private static readonly MAX_POOL_SIZE = 20; // MEMORY LEAK FIX: Reduced buffer pool size to match cache size for consistent memory management\n  private static bufferPoolStats = { hits: 0, misses: 0, created: 0 };\n  \n  private getBuffer(): Buffer {\n    // PERFORMANCE: Track buffer pool usage for optimization monitoring\n    let buffer = DefaultElementProvider.bufferPool.pop();\n    if (buffer) {\n      DefaultElementProvider.bufferPoolStats.hits++;\n      return buffer;\n    } else {\n      DefaultElementProvider.bufferPoolStats.misses++;\n      DefaultElementProvider.bufferPoolStats.created++;\n      buffer = Buffer.alloc(4096);\n      logger.debug(`[DefaultElementProvider] Created new buffer (pool empty), total created: ${DefaultElementProvider.bufferPoolStats.created}`);\n      return buffer;\n    }\n  }\n  \n  private releaseBuffer(buffer: Buffer): void {\n    // CRITICAL FIX: Always attempt to return buffer to pool for reuse\n    if (DefaultElementProvider.bufferPool.length < DefaultElementProvider.MAX_POOL_SIZE) {\n      buffer.fill(0); // SECURITY: Clear buffer before reuse to prevent data leakage\n      DefaultElementProvider.bufferPool.push(buffer);\n    } else {\n      // PERFORMANCE: If pool is full, clear the buffer to help GC\n      buffer.fill(0);\n      logger.debug('[DefaultElementProvider] Buffer pool full, discarding buffer');\n    }\n  }\n\n  /**\n   * Clean up buffer pool and cache to free memory\n   * PERFORMANCE FIX: Added cleanup method to prevent memory leaks\n   * This should be called during application shutdown or periodic cleanup\n   */\n  public static cleanup(): void {\n    // Clear buffer pool\n    DefaultElementProvider.bufferPool.length = 0;\n    \n    // Clear metadata cache\n    DefaultElementProvider.metadataCache.clear();\n    \n    // Clear cached data directory\n    DefaultElementProvider.cachedDataDir = null;\n    \n    // Clear population promises\n    DefaultElementProvider.populateInProgress.clear();\n    \n    logger.info('[DefaultElementProvider] Memory cleanup completed', {\n      bufferStats: DefaultElementProvider.bufferPoolStats,\n      cacheCleared: true\n    });\n    \n    // Reset stats\n    DefaultElementProvider.bufferPoolStats = { hits: 0, misses: 0, created: 0 };\n  }\n\n  /**\n   * Get performance statistics for monitoring\n   * PERFORMANCE MONITORING: Added statistics method for performance tracking\n   * This provides insights into buffer pool efficiency and cache performance\n   * @returns Object containing performance metrics\n   */\n  public static getPerformanceStats(): {\n    bufferPool: {\n      hits: number;\n      misses: number; \n      created: number;\n      hitRate: number;\n      poolSize: number;\n      maxPoolSize: number;\n    };\n    metadataCache: {\n      size: number;\n      maxSize: number;\n    };\n  } {\n    const bufferHits = DefaultElementProvider.bufferPoolStats.hits;\n    const bufferMisses = DefaultElementProvider.bufferPoolStats.misses;\n    const totalRequests = bufferHits + bufferMisses;\n\n    return {\n      bufferPool: {\n        hits: bufferHits,\n        misses: bufferMisses,\n        created: DefaultElementProvider.bufferPoolStats.created,\n        hitRate: totalRequests > 0 ? bufferHits / totalRequests : 0,\n        poolSize: DefaultElementProvider.bufferPool.length,\n        maxPoolSize: DefaultElementProvider.MAX_POOL_SIZE\n      },\n      metadataCache: {\n        size: DefaultElementProvider.metadataCache.size,\n        maxSize: DefaultElementProvider.MAX_CACHE_SIZE\n      }\n    };\n  }\n\n  /**\n   * Read only the YAML frontmatter metadata from a file (first 4KB)\n   *\n   * SECURITY NOTE (Issue #1228): This method MUST use validateContent: true when calling\n   * SecureYamlParser.parse() to ensure Unicode security validation runs. Setting validateContent\n   * to false would disable ALL content validation including zero-width character detection,\n   * creating a security bypass vulnerability.\n   *\n   * @param filePath - Absolute path to the file to read\n   * @param retries - Number of retries for transient failures (default: 2)\n   * @returns Parsed metadata object or null if no valid frontmatter found\n   */\n  private async readMetadataOnly(filePath: string, retries = 2): Promise<any | null> {\n    // PERFORMANCE: Check cache first before reading file\n    try {\n      const stats = await this.statFile(filePath);\n      const cacheKey = filePath;\n      const cached = DefaultElementProvider.metadataCache.get(cacheKey);\n\n      // Return cached metadata if file hasn't changed\n      // CRITICAL FIX: Use integer mtime comparison to avoid floating-point precision issues\n      if (cached && Math.floor(cached.mtime) === Math.floor(stats.mtimeMs) && cached.size === stats.size) {\n        logger.debug(`[DefaultElementProvider] Cache hit for ${filePath}`);\n        return cached.metadata;\n      }\n    } catch {\n      // File doesn't exist, proceed with normal flow\n    }\n    \n    try {\n      // NOTE: Using direct fs.open for partial file read (first 4KB only).\n      // FileOperationsService doesn't support partial reads with file handles.\n      // This is intentionally kept as a direct fs call for performance-critical\n      // metadata extraction that only needs the YAML frontmatter.\n      const { promises: fs } = await import('fs');\n      const fd = await fs.open(filePath, 'r');\n      // PERFORMANCE: Use buffer pool instead of allocating new buffer each time\n      const buffer = this.getBuffer();\n      \n      try {\n        const result = await fd.read(buffer, 0, 4096, 0);\n        const header = buffer.subarray(0, result.bytesRead).toString('utf-8');\n        \n        // Look for YAML frontmatter between --- markers\n        // Support both Unix (\\n) and Windows (\\r\\n) line endings\n        const match = header.match(/^---\\r?\\n([\\s\\S]*?)\\r?\\n---/);\n        if (!match) {\n          return null; // No frontmatter found\n        }\n        \n        // Parse the YAML frontmatter safely\n        try {\n          // SECURITY FIX (Issue #1228): Enable validateContent to block zero-width Unicode and other security threats\n          // SecureYamlParser provides additional validation, injection prevention, and content sanitization\n          // It expects full YAML with --- markers, so we reconstruct the frontmatter block\n          // We disable field-specific validation as this is general metadata parsing, not persona-specific\n          // but we MUST keep validateContent: true to ensure Unicode security validation runs\n          const fullYaml = `---\\n${match[1]}\\n---`;\n          const parseResult = SecureYamlParser.parse(fullYaml, {\n            // REQUIRED: validateContent must be true to enable Unicode security validation including\n            // zero-width character detection (U+200B-U+200F, U+FEFF, etc.). Setting this to false\n            // would disable ALL content validation, creating a security bypass for Unicode-based attacks.\n            // See UnicodeValidator.normalize() for detection logic.\n            validateContent: true,\n            validateFields: false   // Field-specific validation is optional for general metadata parsing\n          });\n          const metadata = parseResult.data;\n          \n          // PERFORMANCE: Cache the metadata with file stats for future reads\n          if (typeof metadata === 'object' && metadata !== null) {\n            try {\n              const stats = await this.statFile(filePath);\n              const cacheEntry: MetadataCacheEntry = {\n                metadata,\n                mtime: stats.mtimeMs,\n                size: stats.size\n              };\n              \n              // CRITICAL MEMORY LEAK FIX: More aggressive cache management to prevent unbounded growth\n              // Check if this entry already exists and just update it instead of adding new\n              if (DefaultElementProvider.metadataCache.has(filePath)) {\n                // Update existing entry - no eviction needed\n                DefaultElementProvider.metadataCache.set(filePath, cacheEntry);\n                logger.debug(`[DefaultElementProvider] Updated existing cache entry for ${filePath}`);\n              } else {\n                // New entry - check if we need to evict first\n                // Use > instead of >= to ensure we never exceed MAX_CACHE_SIZE\n                if (DefaultElementProvider.metadataCache.size >= DefaultElementProvider.MAX_CACHE_SIZE) {\n                  // More aggressive eviction: remove enough entries to stay well under limit\n                  const entriesToEvict = Math.max(1, Math.floor(DefaultElementProvider.MAX_CACHE_SIZE * 0.4));\n                  const keysToEvict = Array.from(DefaultElementProvider.metadataCache.keys()).slice(0, entriesToEvict);\n                  for (const key of keysToEvict) {\n                    DefaultElementProvider.metadataCache.delete(key);\n                  }\n                  logger.debug(`[DefaultElementProvider] Evicted ${keysToEvict.length} cache entries to manage memory (cache size was ${DefaultElementProvider.metadataCache.size + keysToEvict.length})`);\n                }\n                \n                DefaultElementProvider.metadataCache.set(filePath, cacheEntry);\n                logger.debug(`[DefaultElementProvider] Added new cache entry for ${filePath} (cache size now: ${DefaultElementProvider.metadataCache.size})`);\n              }\n            } catch {\n              // Ignore cache errors, return metadata anyway\n            }\n            return metadata;\n          }\n          return null;\n        } catch (yamlError) {\n          // Invalid YAML, return null\n          // ENHANCEMENT: Include error type for better debugging\n          const yamlErrorType = (yamlError as any)?.constructor?.name || 'YAMLError';\n          logger.debug(`[DefaultElementProvider] Invalid YAML in ${filePath}: ${yamlErrorType} - ${yamlError}`);\n          return null;\n        }\n      } finally {\n        // CRITICAL FIX: Ensure file descriptor is closed and buffer is released in ALL paths\n        try {\n          await fd.close();\n        } catch (closeError) {\n          logger.debug(`[DefaultElementProvider] Error closing file descriptor for ${filePath}: ${closeError}`);\n        }\n        // PERFORMANCE: Return buffer to pool for reuse\n        this.releaseBuffer(buffer);\n      }\n    } catch (error: any) {\n      // ENHANCEMENT: Include error type in debug logs for better debugging\n      const errorType = error?.constructor?.name || 'UnknownError';\n      const errorCode = error?.code || 'NO_CODE';\n      \n      // RELIABILITY: Add retry logic for transient failures\n      if (retries > 0 && (errorCode === 'EBUSY' || errorCode === 'EAGAIN')) {\n        logger.debug(`[DefaultElementProvider] Retrying read for ${filePath} after ${errorType}:${errorCode}`);\n        await new Promise(resolve => setTimeout(resolve, 50)); // Brief delay before retry\n        return this.readMetadataOnly(filePath, retries - 1);\n      }\n      \n      logger.debug(`[DefaultElementProvider] Could not read metadata from ${filePath}: ${errorType}:${errorCode} - ${error?.message || error}`);\n      return null;\n    }\n  }\n\n  /**\n   * Check if a file is a DollhouseMCP test element based on metadata\n   * This replaces filename pattern detection with accurate metadata-based detection\n   * @param filePath Path to the file to check\n   * @returns true if the file contains _dollhouseMCPTest: true metadata\n   */\n  private async isDollhouseMCPTestElement(filePath: string): Promise<boolean> {\n    try {\n      const metadata = await this.readMetadataOnly(filePath);\n      const isTest = !!(metadata && metadata._dollhouseMCPTest === true);\n      \n      \n      return isTest;\n    } catch (error) {\n      // If we can't read the metadata, assume it's not a test file\n      logger.debug(`[DefaultElementProvider] Error checking test metadata for ${filePath}: ${error}`);\n      return false;\n    }\n  }\n\n  /**\n   * Detect if we're in a production environment by checking for production indicators\n   * Uses a confidence-based approach requiring multiple indicators for better accuracy\n   * @returns true if this appears to be a production environment\n   */\n  private isProductionEnvironment(): boolean {\n    // Allow tests to explicitly override production mode detection\n    if (process.env.FORCE_PRODUCTION_MODE === 'true') {\n      return true;\n    }\n    if (process.env.FORCE_PRODUCTION_MODE === 'false') {\n      return false;\n    }\n    \n    // Weighted indicators for production detection\n    const indicators = {\n      // Strong indicators (weight: 2)\n      hasUserHomeDir: (process.env.HOME && (process.env.HOME.includes('/Users/') || process.env.HOME.includes('/home/'))) || \n                      !!process.env.USERPROFILE,\n      isProductionNode: process.env.NODE_ENV === 'production',\n      notInTestDir: (() => {\n        const cwd = process.cwd().toLowerCase();\n        // Normalize path separators for cross-platform checking (Windows uses \\ but checks use /)\n        const normalizedCwd = cwd.replaceAll('\\\\', '/');\n        return !normalizedCwd.includes('/test') && \n               !normalizedCwd.includes('/__tests__') && \n               !normalizedCwd.includes('/temp') &&\n               !normalizedCwd.includes('/dist/test');\n      })(),\n      \n      // Moderate indicators (weight: 1)\n      notInCI: !process.env.CI,\n      noTestEnv: process.env.NODE_ENV !== 'test',\n      noDevEnv: process.env.NODE_ENV !== 'development',\n    };\n    \n    // Calculate weighted score\n    let score = 0;\n    if (indicators.hasUserHomeDir) score += 2;\n    if (indicators.isProductionNode) score += 2;\n    if (indicators.notInTestDir) score += 2;\n    if (indicators.notInCI) score += 1;\n    if (indicators.noTestEnv) score += 1;\n    if (indicators.noDevEnv) score += 1;\n    \n    // Log detection details for debugging\n    const activeIndicators = Object.entries(indicators)\n      .filter(([_, value]) => value)\n      .map(([key]) => key);\n    \n    // TYPESCRIPT FIX: Removed logger.isDebugEnabled() check as this method doesn't exist on MCPLogger\n    // The logger already handles debug level internally, so we can call debug() directly\n    if (score >= 3) {\n      logger.debug(\n        '[DefaultElementProvider] Production environment detected',\n        { score, activeIndicators, forceMode: 'not set' }\n      );\n    }\n    \n    // Require a score of at least 3 for production detection (more confident)\n    // This prevents false positives in edge cases while maintaining security\n    return score >= 3;\n  }\n  \n  /**\n   * Copy all files from source directory to destination directory\n   * Skips files that already exist to preserve user modifications\n   */\n  private async copyElementFiles(sourceDir: string, destDir: string, elementType: string): Promise<number> {\n    let copiedCount = 0;\n    \n    \n    try {\n      // Ensure destination directory exists\n      await this.makeDirectory(destDir);\n\n      // Read source directory\n      const files = await this.listDir(sourceDir);\n      \n      \n      for (const file of files) {\n        \n        // Only copy markdown files\n        if (!file.endsWith(FILE_CONSTANTS.ELEMENT_EXTENSION)) {\n          continue;\n        }\n        \n        // Normalize filename for security\n        const normalizedFile = UnicodeValidator.normalize(file);\n        if (!normalizedFile.isValid) {\n          logger.warn(`[DefaultElementProvider] Skipping file with invalid Unicode: ${file}`);\n          continue;\n        }\n\n        const sourcePath = path.join(sourceDir, normalizedFile.normalizedContent);\n        const destPath = path.join(destDir, normalizedFile.normalizedContent);\n        \n        \n        // SECURITY FIX: Validate file paths to prevent path traversal attacks\n        // This prevents malicious files from escaping the intended directory structure\n        const sourceValid = this.validateFilePath(sourcePath, [sourceDir]);\n        const destValid = this.validateFilePath(destPath, [destDir]);\n        \n        if (!sourceValid || !destValid) {\n          logger.warn(\n            `[DefaultElementProvider] Skipping file with invalid path: ${normalizedFile.normalizedContent}`,\n            { sourcePath, destPath, elementType }\n          );\n          continue;\n        }\n        \n        // Production safety check: Block DollhouseMCP test elements in production environments\n        // Skip this check if loadTestData is explicitly enabled (for testing scenarios)\n        if (!this.config.loadTestData && this.isProductionEnvironment()) {\n          const isDollhouseTest = await this.isDollhouseMCPTestElement(sourcePath);\n          \n          if (isDollhouseTest) {\n            logger.warn(\n              `[DefaultElementProvider] SECURITY: Blocking DollhouseMCP test element in production: ${normalizedFile.normalizedContent}`,\n              { \n                file: normalizedFile.normalizedContent,\n                reason: 'DollhouseMCP test element detected in production environment',\n                elementType\n              }\n            );\n            \n            // Log security event for blocked test data\n            SecurityMonitor.logSecurityEvent({\n              type: 'TEST_DATA_BLOCKED',\n              severity: 'MEDIUM',\n              source: 'DefaultElementProvider.copyElementFiles',\n              details: `Blocked DollhouseMCP test element in production: ${normalizedFile.normalizedContent}`,\n              metadata: {\n                filename: normalizedFile.normalizedContent,\n                elementType,\n                reason: 'DollhouseMCP test element detected in production environment',\n                detectionMethod: 'metadata-based'\n              }\n            });\n            \n            continue;\n          }\n        }\n        \n        // Check if destination file already exists\n        const destExists = await this.fileExists(destPath);\n        if (destExists) {\n          logger.debug(`[DefaultElementProvider] Skipping existing file: ${normalizedFile.normalizedContent}`);\n          continue;\n        }\n        \n        try {\n          // Validate source file before copying\n          const sourceStats = await this.statFile(sourcePath);\n\n          // Check file size limit\n          if (sourceStats.size > FILE_CONSTANTS.MAX_FILE_SIZE) {\n            logger.warn(\n              `[DefaultElementProvider] Skipping oversized file ${normalizedFile.normalizedContent}: ` +\n              `${sourceStats.size} bytes (max: ${FILE_CONSTANTS.MAX_FILE_SIZE} bytes)`,\n              { \n                file: normalizedFile.normalizedContent, \n                size: sourceStats.size,\n                maxSize: FILE_CONSTANTS.MAX_FILE_SIZE,\n                elementType \n              }\n            );\n            continue;\n          }\n          \n          // Copy the file with verification\n          \n          await this.copyFileWithVerification(sourcePath, destPath);\n          copiedCount++;\n          \n          logger.debug(`[DefaultElementProvider] Copied ${elementType}: ${normalizedFile.normalizedContent}`);\n          \n          // Log security event for each file copied\n          SecurityMonitor.logSecurityEvent({\n            type: 'FILE_COPIED',\n            severity: 'LOW',\n            source: 'DefaultElementProvider.copyElementFiles',\n            details: `Copied default ${elementType} file: ${normalizedFile.normalizedContent}`,\n            metadata: {\n              sourcePath,\n              destPath,\n              elementType,\n              fileSize: (await this.statFile(destPath)).size\n            }\n          });\n        } catch (error) {\n          const err = error as Error;\n          logger.error(\n            `[DefaultElementProvider] Failed to copy ${normalizedFile.normalizedContent}`,\n            { \n              error: err.message,\n              stack: err.stack,\n              sourcePath,\n              destPath,\n              elementType\n            }\n          );\n          // Continue with other files instead of failing completely\n        }\n      }\n      \n      if (copiedCount > 0) {\n        logger.info(`[DefaultElementProvider] Copied ${copiedCount} ${elementType} file(s)`);\n      }\n      \n    } catch (error) {\n      logger.error(`[DefaultElementProvider] Error copying ${elementType} files:`, error);\n    }\n    \n    return copiedCount;\n  }\n  \n  /**\n   * Copy a file with integrity verification\n   * Ensures the file was copied correctly by comparing sizes\n   */\n  /**\n   * Calculate checksum of a file for integrity verification\n   * @param filePath Path to the file\n   * @returns Hex-encoded checksum\n   */\n  private async calculateChecksum(filePath: string): Promise<string> {\n    const hash = createHash(FILE_CONSTANTS.CHECKSUM_ALGORITHM);\n    // NOTE: Using direct fs.open for chunked/streaming file read.\n    // FileOperationsService doesn't support file handles for chunked reading.\n    // This is intentionally kept as a direct fs call for memory-efficient\n    // checksum calculation on potentially large files.\n    const { promises: fs } = await import('fs');\n    const stream = await fs.open(filePath, 'r');\n\n    try {\n      const buffer = Buffer.alloc(FILE_CONSTANTS.CHUNK_SIZE);\n      let bytesRead: number;\n\n      do {\n        const result = await stream.read(buffer, 0, FILE_CONSTANTS.CHUNK_SIZE);\n        bytesRead = result.bytesRead;\n\n        if (bytesRead > 0) {\n          hash.update(buffer.subarray(0, bytesRead));\n        }\n      } while (bytesRead > 0);\n\n      return hash.digest('hex');\n    } finally {\n      await stream.close();\n    }\n  }\n\n  /**\n   * Copy file with integrity verification and retry logic\n   * @param sourcePath Source file path\n   * @param destPath Destination file path\n   * @throws Error if copy fails after all retry attempts\n   */\n  private async copyFileWithVerification(sourcePath: string, destPath: string): Promise<void> {\n    let lastError: Error | null = null;\n\n    for (let attempt = 1; attempt <= COPY_RETRY_ATTEMPTS; attempt++) {\n      try {\n        // Copy the file\n        await this.copyFileOp(sourcePath, destPath);\n\n        // Verify size matches\n        const [sourceStats, destStats] = await Promise.all([\n          this.statFile(sourcePath),\n          this.statFile(destPath)\n        ]);\n\n        if (sourceStats.size !== destStats.size) {\n          throw new Error(\n            `Size mismatch after copy - source: ${sourceStats.size} bytes, ` +\n            `destination: ${destStats.size} bytes`\n          );\n        }\n\n        // Verify checksum matches for complete integrity\n        const [sourceChecksum, destChecksum] = await Promise.all([\n          this.calculateChecksum(sourcePath),\n          this.calculateChecksum(destPath)\n        ]);\n\n        if (sourceChecksum !== destChecksum) {\n          throw new Error(\n            `Checksum mismatch after copy - source: ${sourceChecksum}, ` +\n            `destination: ${destChecksum}`\n          );\n        }\n\n        // Set proper permissions\n        try {\n          await this.chmodFile(destPath, FILE_CONSTANTS.FILE_PERMISSIONS);\n        } catch (error) {\n          logger.debug(\n            `[DefaultElementProvider] Could not set permissions on ${destPath}: ${error}`,\n            { sourcePath, destPath, attempt }\n          );\n        }\n\n        // Success - file copied and verified\n        logger.debug(\n          `[DefaultElementProvider] Successfully copied and verified: ${path.basename(sourcePath)}`,\n          { size: sourceStats.size, checksum: sourceChecksum.substring(0, 8) }\n        );\n        return;\n\n      } catch (error) {\n        lastError = error as Error;\n\n        // Clean up failed copy\n        try {\n          await this.deleteFileOp(destPath);\n        } catch {\n          // Ignore cleanup errors\n        }\n\n        if (attempt < COPY_RETRY_ATTEMPTS) {\n          logger.debug(\n            `[DefaultElementProvider] Copy attempt ${attempt} failed, retrying...`,\n            { error: lastError.message, sourcePath, destPath }\n          );\n          await new Promise(resolve => setTimeout(resolve, COPY_RETRY_DELAY * attempt));\n        }\n      }\n    }\n    \n    // All attempts failed\n    throw new Error(\n      `Failed to copy ${path.basename(sourcePath)} after ${COPY_RETRY_ATTEMPTS} attempts: ` +\n      `${lastError?.message || 'Unknown error'}`\n    );\n  }\n  \n  /**\n   * Populate the portfolio with default elements from bundled data\n   * This is called during portfolio initialization for new installations\n   * Protected against concurrent calls to prevent race conditions\n   */\n  public async populateDefaults(portfolioBaseDir: string): Promise<void> {\n    // Check if population is already in progress for this portfolio\n    const existingPopulation = DefaultElementProvider.populateInProgress.get(portfolioBaseDir);\n    if (existingPopulation) {\n      logger.debug(\n        '[DefaultElementProvider] Population already in progress for portfolio, waiting...',\n        { portfolioBaseDir }\n      );\n      return existingPopulation;\n    }\n    \n    // Create new population promise\n    const populationPromise = this.performPopulation(portfolioBaseDir)\n      .finally(() => {\n        // Clean up when done\n        DefaultElementProvider.populateInProgress.delete(portfolioBaseDir);\n      });\n    \n    DefaultElementProvider.populateInProgress.set(portfolioBaseDir, populationPromise);\n    return populationPromise;\n  }\n  \n  /**\n   * Perform the actual population of default elements\n   * @param portfolioBaseDir Base directory of the portfolio\n   */\n  private async performPopulation(portfolioBaseDir: string): Promise<void> {\n    // Check if test data loading is disabled\n    // Note: This check is needed even though constructor sets config, because\n    // config can be overridden after construction\n    \n    // Use production environment detection that respects FORCE_PRODUCTION_MODE\n    const isDevelopmentMode = !this.isProductionEnvironment();\n    \n    if (isDevelopmentMode && !this.config.loadTestData) {\n      logger.info(\n        '[DefaultElementProvider] Skipping default element population in development mode',\n        { \n          portfolioBaseDir,\n          reason: 'Test data loading disabled',\n          enableWith: 'Set DOLLHOUSE_LOAD_TEST_DATA=true to enable'\n        }\n      );\n      return;\n    }\n    \n    logger.info(\n      '[DefaultElementProvider] Starting default element population',\n      { portfolioBaseDir }\n    );\n    \n    // Log security event for portfolio initialization\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_INITIALIZATION',\n      severity: 'LOW',\n      source: 'DefaultElementProvider.performPopulation',\n      details: `Starting default element population for portfolio: ${portfolioBaseDir}`\n    });\n    \n    // Find the bundled data directory\n    const dataDir = await this.findDataDirectory();\n    if (!dataDir) {\n      logger.warn(\n        '[DefaultElementProvider] No bundled data directory found - portfolio will start empty',\n        { \n          searchPaths: this.dataSearchPaths.slice(0, 3), // Log first few paths for debugging\n          cwd: process.cwd(),\n          dirname: this.__dirname\n        }\n      );\n      return;\n    }\n    \n    // Track total files copied\n    let totalCopied = 0;\n    const copiedCounts: Record<string, number> = {};\n    \n    // Copy each element type - directories now match enum values (all plural)\n    for (const elementType of Object.values(ElementType)) {\n      const sourceDir = path.join(dataDir, elementType);\n      const destDir = path.join(portfolioBaseDir, elementType);\n\n      // Check if source directory exists\n      const sourceDirExists = await this.fileExists(sourceDir);\n      if (!sourceDirExists) {\n        // Source directory doesn't exist, skip\n        logger.debug(`[DefaultElementProvider] No ${elementType} directory in bundled data`);\n        continue;\n      }\n\n      try {\n        const copiedCount = await this.copyElementFiles(sourceDir, destDir, elementType);\n        copiedCounts[elementType] = copiedCount;\n        totalCopied += copiedCount;\n      } catch (error) {\n        logger.error(`[DefaultElementProvider] Error copying ${elementType} elements:`, error);\n      }\n    }\n    \n    if (totalCopied > 0) {\n      logger.info(\n        `[DefaultElementProvider] Successfully populated portfolio with ${totalCopied} default element(s)`,\n        {\n          portfolioBaseDir,\n          dataDir,\n          breakdown: copiedCounts\n        }\n      );\n      \n      // Log security event for successful population\n      SecurityMonitor.logSecurityEvent({\n        type: 'PORTFOLIO_POPULATED',\n        severity: 'LOW',\n        source: 'DefaultElementProvider.performPopulation',\n        details: `Successfully populated portfolio with ${totalCopied} default elements`,\n        metadata: {\n          portfolioBaseDir,\n          dataDir,\n          copiedCounts\n        }\n      });\n    } else {\n      logger.info('[DefaultElementProvider] No new elements to copy - portfolio may already have content');\n    }\n  }\n}"]}
@@ -34,6 +34,21 @@ export interface ContentValidatorOptions {
34
34
  contentContext?: 'persona' | 'skill' | 'template' | 'agent' | 'memory';
35
35
  }
36
36
  export declare class ContentValidator {
37
+ /**
38
+ * SHA-256 hashes of bundled data/ elements verified against HASHES.json at seed time.
39
+ * Content whose hash is in this set bypasses injection-pattern checks — it was
40
+ * vetted by the DollhouseMCP team before being included in the npm package.
41
+ * Unicode and YAML-bomb checks still run regardless.
42
+ *
43
+ * Populated by DefaultElementProvider.registerBundledHashes() on startup.
44
+ * Only content that matches the published HASHES.json is registered — any
45
+ * modification to a bundled file after install breaks the hash and revokes trust.
46
+ */
47
+ private static readonly bundledContentHashes;
48
+ /** Register a SHA-256 hex hash as trusted bundled content. */
49
+ static registerBundledHash(sha256hex: string): void;
50
+ /** True if the given content hash belongs to a verified bundled element. */
51
+ static isBundledContent(content: string): boolean;
37
52
  private static telemetryResolver?;
38
53
  static configureTelemetryResolver(resolver: () => SecurityTelemetry | undefined): void;
39
54
  private static getTelemetry;
@@ -1 +1 @@
1
- {"version":3,"file":"contentValidator.d.ts","sourceRoot":"","sources":["../../src/security/contentValidator.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAQH,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;IAIH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,kBAAkB,CAwExC;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;IAiErC;;;;;;;;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"}
1
+ {"version":3,"file":"contentValidator.d.ts","sourceRoot":"","sources":["../../src/security/contentValidator.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AASH,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;;;;;;;;;OASG;IACH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,oBAAoB,CAAqB;IAEjE,8DAA8D;IAC9D,MAAM,CAAC,mBAAmB,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI;IAInD,4EAA4E;IAC5E,MAAM,CAAC,gBAAgB,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO;IAMjD,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;IAIH,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,kBAAkB,CAwExC;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;IAiErC;;;;;;;;OAQG;IACH,MAAM,CAAC,mBAAmB,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,GAAE,uBAA4B,GAAG,uBAAuB;IAiF3G;;;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"}
@@ -6,6 +6,7 @@
6
6
  *
7
7
  * Security: SEC-001 - Critical vulnerability protection
8
8
  */
9
+ import { createHash } from 'node:crypto';
9
10
  import { SecurityError } from './errors.js';
10
11
  import { SecurityMonitor } from './securityMonitor.js';
11
12
  import { logger } from '../utils/logger.js';
@@ -13,6 +14,28 @@ import { RegexValidator } from './regexValidator.js';
13
14
  import { SECURITY_LIMITS } from './constants.js';
14
15
  import { UnicodeValidator } from './validators/unicodeValidator.js';
15
16
  export class ContentValidator {
17
+ /**
18
+ * SHA-256 hashes of bundled data/ elements verified against HASHES.json at seed time.
19
+ * Content whose hash is in this set bypasses injection-pattern checks — it was
20
+ * vetted by the DollhouseMCP team before being included in the npm package.
21
+ * Unicode and YAML-bomb checks still run regardless.
22
+ *
23
+ * Populated by DefaultElementProvider.registerBundledHashes() on startup.
24
+ * Only content that matches the published HASHES.json is registered — any
25
+ * modification to a bundled file after install breaks the hash and revokes trust.
26
+ */
27
+ static bundledContentHashes = new Set();
28
+ /** Register a SHA-256 hex hash as trusted bundled content. */
29
+ static registerBundledHash(sha256hex) {
30
+ this.bundledContentHashes.add(sha256hex);
31
+ }
32
+ /** True if the given content hash belongs to a verified bundled element. */
33
+ static isBundledContent(content) {
34
+ if (this.bundledContentHashes.size === 0)
35
+ return false;
36
+ const hash = createHash('sha256').update(content).digest('hex');
37
+ return this.bundledContentHashes.has(hash);
38
+ }
16
39
  static telemetryResolver;
17
40
  static configureTelemetryResolver(resolver) {
18
41
  this.telemetryResolver = resolver;
@@ -101,7 +124,7 @@ export class ContentValidator {
101
124
  { pattern: /<object[\s>]/gi, severity: 'high', description: 'HTML object injection' },
102
125
  { pattern: /<embed[\s>]/gi, severity: 'high', description: 'HTML embed injection' },
103
126
  { pattern: /\bon\w+=\s*["']/gi, severity: 'critical', description: 'HTML event handler injection' },
104
- { pattern: /javascript\s*:/gi, severity: 'critical', description: 'JavaScript protocol injection' },
127
+ { pattern: /javascript[ \t]*:[ \t]*\S/gi, severity: 'critical', description: 'JavaScript protocol injection' },
105
128
  // Entity-encoded variants: &#106;avascript, &#x6a;avascript, &#106;&#97;vascript, etc.
106
129
  { 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' },
107
130
  // Fully/partially entity-encoded: detects &#...script pattern (covers multi-entity encoding)
@@ -379,6 +402,21 @@ export class ContentValidator {
379
402
  throw new SecurityError(`Content exceeds maximum length of ${maxLength} characters after normalization (${unicodeCheck.sanitized.length} provided)`);
380
403
  }
381
404
  }
405
+ // Skip injection-pattern scanning for verified bundled elements.
406
+ // Content whose SHA-256 matches HASHES.json was reviewed by the DollhouseMCP
407
+ // team before being included in the npm package — false positives from
408
+ // legitimate YAML keys (javascript:) or educational payloads (wget in pentest
409
+ // templates) should not fire CRITICAL alerts at every install.
410
+ // Unicode and YAML-bomb checks above still run unconditionally.
411
+ if (this.isBundledContent(content)) {
412
+ logger.debug('[ContentValidator] Skipping injection scan for verified bundled element');
413
+ return {
414
+ isValid: true,
415
+ sanitizedContent: unicodeCheck.sanitized,
416
+ detectedPatterns: [],
417
+ severity: 'low'
418
+ };
419
+ }
382
420
  // Check for injection patterns on ORIGINAL content (to catch encoded attacks)
383
421
  // but apply replacements to NORMALIZED content (to preserve normalization)
384
422
  const injectionCheck = this.checkInjectionPatterns(content, unicodeCheck.sanitized, detectedPatterns, unicodeCheck.highestSeverity, maxLength, options.contentContext);
@@ -606,4 +644,4 @@ export class ContentValidator {
606
644
  return `---\n${yamlContent}\n---${contentResult.sanitizedContent || markdownContent}`;
607
645
  }
608
646
  }
609
- //# 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 { logger } from '../utils/logger.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  // MAINTENANCE: Review periodically for false positives (run integration tests in\n  // tests/integration/security-audit-batch-a.integration.test.ts) and new attack vectors.\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+(in\\s+)?(admin|root|system|sudo|developer|debug|test|DAN)\\s*(mode)?/gi, severity: 'critical', description: 'Role elevation attempt' },\n    // Specific dangerous roles only — \"act as \\w+\" would false-positive on persona\n    // content like \"act as a helpful teacher\" (#1782-4, review feedback)\n    { pattern: /act\\s+as\\s+(admin|root|system|sudo|superuser|DAN)\\b/gi, severity: 'critical', description: 'Role elevation attempt' },\n    { pattern: /pretend\\s+you\\s+have\\s+no\\s+(guidelines|restrictions|rules|limits)/gi, severity: 'critical', description: 'Guideline bypass attempt' },\n    { pattern: /\\b(jailbreak|do\\s+anything\\s+now|DAN\\s+mode)\\b/gi, severity: 'critical', description: 'Jailbreak 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]{1,500}/gi, severity: 'critical', description: 'External command execution' },\n    { pattern: /wget\\s+[^\\s]{1,500}/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        logger.debug(`Content injection blocked: ${description} (${severity}) — pattern: ${pattern.source}`);\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}"]}
647
+ //# 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 { createHash } from 'node:crypto';\nimport { SecurityError } from './errors.js';\nimport { SecurityMonitor } from './securityMonitor.js';\nimport { logger } from '../utils/logger.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  /**\n   * SHA-256 hashes of bundled data/ elements verified against HASHES.json at seed time.\n   * Content whose hash is in this set bypasses injection-pattern checks — it was\n   * vetted by the DollhouseMCP team before being included in the npm package.\n   * Unicode and YAML-bomb checks still run regardless.\n   *\n   * Populated by DefaultElementProvider.registerBundledHashes() on startup.\n   * Only content that matches the published HASHES.json is registered — any\n   * modification to a bundled file after install breaks the hash and revokes trust.\n   */\n  private static readonly bundledContentHashes = new Set<string>();\n\n  /** Register a SHA-256 hex hash as trusted bundled content. */\n  static registerBundledHash(sha256hex: string): void {\n    this.bundledContentHashes.add(sha256hex);\n  }\n\n  /** True if the given content hash belongs to a verified bundled element. */\n  static isBundledContent(content: string): boolean {\n    if (this.bundledContentHashes.size === 0) return false;\n    const hash = createHash('sha256').update(content).digest('hex');\n    return this.bundledContentHashes.has(hash);\n  }\n\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  // MAINTENANCE: Review periodically for false positives (run integration tests in\n  // tests/integration/security-audit-batch-a.integration.test.ts) and new attack vectors.\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+(in\\s+)?(admin|root|system|sudo|developer|debug|test|DAN)\\s*(mode)?/gi, severity: 'critical', description: 'Role elevation attempt' },\n    // Specific dangerous roles only — \"act as \\w+\" would false-positive on persona\n    // content like \"act as a helpful teacher\" (#1782-4, review feedback)\n    { pattern: /act\\s+as\\s+(admin|root|system|sudo|superuser|DAN)\\b/gi, severity: 'critical', description: 'Role elevation attempt' },\n    { pattern: /pretend\\s+you\\s+have\\s+no\\s+(guidelines|restrictions|rules|limits)/gi, severity: 'critical', description: 'Guideline bypass attempt' },\n    { pattern: /\\b(jailbreak|do\\s+anything\\s+now|DAN\\s+mode)\\b/gi, severity: 'critical', description: 'Jailbreak 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]{1,500}/gi, severity: 'critical', description: 'External command execution' },\n    { pattern: /wget\\s+[^\\s]{1,500}/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[ \\t]*:[ \\t]*\\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        logger.debug(`Content injection blocked: ${description} (${severity}) — pattern: ${pattern.source}`);\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    // Skip injection-pattern scanning for verified bundled elements.\n    // Content whose SHA-256 matches HASHES.json was reviewed by the DollhouseMCP\n    // team before being included in the npm package — false positives from\n    // legitimate YAML keys (javascript:) or educational payloads (wget in pentest\n    // templates) should not fire CRITICAL alerts at every install.\n    // Unicode and YAML-bomb checks above still run unconditionally.\n    if (this.isBundledContent(content)) {\n      logger.debug('[ContentValidator] Skipping injection scan for verified bundled element');\n      return {\n        isValid: true,\n        sanitizedContent: unicodeCheck.sanitized,\n        detectedPatterns: [],\n        severity: 'low'\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}"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dollhousemcp/mcp-server",
3
- "version": "2.0.12",
3
+ "version": "2.0.13",
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",
@@ -10,7 +10,7 @@
10
10
  "dollhouse-console-token": "dist/cli/console-token.js"
11
11
  },
12
12
  "scripts": {
13
- "prebuild": "npm run build:safety && node scripts/generate-version.js",
13
+ "prebuild": "npm run build:safety && node scripts/generate-version.js && node scripts/generate-hashes.mjs",
14
14
  "build:safety": "cd packages/safety && npm install && npm run build",
15
15
  "build": "tsc",
16
16
  "postbuild": "node -e \"const fs=require('fs'),p=require('path');function rm(d){if(fs.existsSync(d)){for(const e of fs.readdirSync(d)){const f=p.join(d,e);fs.statSync(f).isDirectory()?rm(f):fs.unlinkSync(f)}fs.rmdirSync(d)}}function cpR(s,d){fs.mkdirSync(d,{recursive:true});for(const e of fs.readdirSync(s)){const sp=p.join(s,e),dp=p.join(d,e);fs.statSync(sp).isDirectory()?cpR(sp,dp):fs.copyFileSync(sp,dp)}}cpR('src/seed-elements','dist/seed-elements');rm('dist/web/public');cpR('src/web/public','dist/web/public')\"",
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.12",
6
+ "version": "2.0.13",
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.12",
32
+ "version": "2.0.13",
33
33
  "transport": {
34
34
  "type": "stdio"
35
35
  }