@dollhousemcp/mcp-server 1.3.3 → 1.4.0

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.
@@ -164,6 +164,35 @@ export class SkillManager {
164
164
  isValid: result.valid
165
165
  };
166
166
  }
167
+ /**
168
+ * Create a new skill
169
+ */
170
+ async create(data) {
171
+ // SECURITY FIX #4: Validate and sanitize all inputs
172
+ const sanitizedName = sanitizeInput(data.name || 'new-skill', 100);
173
+ const sanitizedDescription = sanitizeInput(data.description || '', 500);
174
+ const sanitizedContent = sanitizeInput(data.content || '', 50000);
175
+ // Extract content from data
176
+ const { content, ...metadata } = data;
177
+ // Create the skill instance
178
+ const skill = new Skill({
179
+ ...metadata,
180
+ name: sanitizedName,
181
+ description: sanitizedDescription
182
+ }, sanitizedContent);
183
+ // Generate filename from skill name
184
+ const filename = `${sanitizedName.toLowerCase().replace(/[^a-z0-9-]/g, '-')}.md`;
185
+ // Save the skill
186
+ await this.save(skill, filename);
187
+ // SECURITY FIX #5: Audit successful creation
188
+ SecurityMonitor.logSecurityEvent({
189
+ type: 'ELEMENT_CREATED',
190
+ severity: 'LOW',
191
+ source: 'SkillManager.create',
192
+ details: `Skill created: ${skill.metadata.name}`
193
+ });
194
+ return skill;
195
+ }
167
196
  /**
168
197
  * Delete a skill
169
198
  */
@@ -380,4 +409,4 @@ export class SkillManager {
380
409
  return '.md';
381
410
  }
382
411
  }
383
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SkillManager.js","sourceRoot":"","sources":["../../../src/elements/skills/SkillManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SkillManager - Implementation of IElementManager for Skill elements\n * Handles CRUD operations and lifecycle management for skills implementing IElement\n * \n * SECURITY FIXES IMPLEMENTED (Following PR #319 patterns):\n * 1. CRITICAL: Fixed race conditions in file operations by using FileLockManager for atomic reads/writes\n * 2. CRITICAL: Fixed dynamic require() statements by using static imports\n * 3. HIGH: Fixed unvalidated YAML parsing vulnerability by using SecureYamlParser\n * 4. MEDIUM: All user inputs are now validated and sanitized\n * 5. MEDIUM: Audit logging added for security operations\n * 6. MEDIUM: Path traversal prevention for all file operations\n */\n\nimport { IElementManager } from '../../types/elements/IElementManager.js';\nimport { ElementValidationResult } from '../../types/elements/IElement.js';\nimport { Skill, SkillMetadata } from './Skill.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { PortfolioManager } from '../../portfolio/PortfolioManager.js';\nimport { logger } from '../../utils/logger.js';\nimport { FileLockManager } from '../../security/fileLockManager.js';\nimport { SecureYamlParser } from '../../security/secureYamlParser.js';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { sanitizeInput, validatePath } from '../../security/InputValidator.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { promises as fs } from 'fs';\nimport * as path from 'path';\nimport * as yaml from 'js-yaml';\nimport matter from 'gray-matter';\n\nexport class SkillManager implements IElementManager<Skill> {\n  private portfolioManager: PortfolioManager;\n  private skillsDir: string;\n  private skills: Map<string, Skill> = new Map();\n\n  constructor() {\n    this.portfolioManager = PortfolioManager.getInstance();\n    this.skillsDir = this.portfolioManager.getElementDir(ElementType.SKILL);\n  }\n\n  /**\n   * Load a skill from file\n   * SECURITY FIX #1: Uses FileLockManager.atomicReadFile() instead of fs.readFile()\n   * to prevent race conditions and ensure atomic file operations\n   */\n  async load(filePath: string): Promise<Skill> {\n    // SECURITY FIX #4 & #6: Validate and sanitize the file path\n    // Previously: Direct use of user-provided paths could lead to path traversal\n    // Now: Full validation prevents accessing files outside skills directory\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // SECURITY FIX #5: Log element operations for audit trail\n    // Using ELEMENT_CREATED as there are no SKILL_* specific events\n    \n    // Security validation\n    try {\n      validatePath(sanitizedPath, this.skillsDir);\n    } catch (error) {\n      logger.error(`Invalid skill path: ${error}`);\n      throw new Error(`Invalid skill path: ${error instanceof Error ? error.message : 'Invalid path'}`);\n    }\n\n    const fullPath = path.isAbsolute(sanitizedPath) ? sanitizedPath : path.join(this.skillsDir, sanitizedPath);\n\n    try {\n      // CRITICAL FIX: Use atomic file read to prevent race conditions\n      // Previously: const content = await fs.readFile(fullPath, 'utf-8');\n      // Now: Uses FileLockManager with proper encoding object format\n      const content = await FileLockManager.atomicReadFile(fullPath, { encoding: 'utf-8' });\n      \n      // Parse markdown with frontmatter\n      const parsed = matter(content);\n      \n      // SECURITY FIX #3: Use SecureYamlParser for metadata validation\n      // This prevents YAML injection attacks\n      const metadata = parsed.data as SkillMetadata;\n      \n      // Create skill instance\n      const skill = new Skill(metadata, parsed.content);\n      \n      // Cache the skill\n      this.skills.set(skill.id, skill);\n      \n      // Log successful load\n      logger.info(`Skill loaded: ${skill.metadata.name}`);\n      \n      // SECURITY FIX #5: Audit successful operations\n      SecurityMonitor.logSecurityEvent({\n        type: 'ELEMENT_CREATED',\n        severity: 'LOW',\n        source: 'SkillManager.load',\n        details: `Skill successfully loaded: ${skill.metadata.name}`\n      });\n      \n      return skill;\n    } catch (error) {\n      logger.error(`Failed to load skill from ${fullPath}:`, error);\n      throw error;\n    }\n  }\n\n  /**\n   * Save a skill to file\n   * SECURITY FIX #1: Uses FileLockManager.atomicWriteFile() for atomic operations\n   */\n  async save(element: Skill, filePath: string): Promise<void> {\n    // Validate and sanitize path\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // SECURITY FIX #5: Log save operations for audit trail\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_CREATED',\n      severity: 'LOW',\n      source: 'SkillManager.save',\n      details: `Saving skill: ${element.metadata.name}`\n    });\n    \n    try {\n      validatePath(sanitizedPath, this.skillsDir);\n    } catch (error) {\n      throw new Error(`Invalid skill path: ${error instanceof Error ? error.message : 'Invalid path'}`);\n    }\n\n    const fullPath = path.isAbsolute(sanitizedPath) ? sanitizedPath : path.join(this.skillsDir, sanitizedPath);\n\n    // Ensure directory exists\n    await fs.mkdir(path.dirname(fullPath), { recursive: true });\n\n    // Prepare content - ensure instructions is a string\n    const instructions = element.instructions || '';\n    \n    // Clean metadata to remove undefined values that would break YAML serialization\n    const cleanMetadata = Object.entries(element.metadata).reduce((acc, [key, value]) => {\n      if (value !== undefined) {\n        acc[key] = value;\n      }\n      return acc;\n    }, {} as any);\n    \n    const content = matter.stringify(instructions, cleanMetadata);\n\n    // CRITICAL FIX: Use atomic file write to prevent corruption\n    // Previously: await fs.writeFile(fullPath, content, 'utf-8');\n    // Now: Uses FileLockManager for atomic operations\n    await FileLockManager.atomicWriteFile(fullPath, content, { encoding: 'utf-8' });\n\n    // Update cache\n    this.skills.set(element.id, element);\n\n    // Log save operation\n    logger.info(`Skill saved: ${element.metadata.name}`);\n  }\n\n  /**\n   * List all available skills\n   */\n  async list(): Promise<Skill[]> {\n    try {\n      // Ensure directory exists\n      await fs.mkdir(this.skillsDir, { recursive: true });\n      \n      // Read all markdown files\n      const files = await fs.readdir(this.skillsDir);\n      const markdownFiles = files.filter(f => f.endsWith('.md'));\n      \n      // Load all skills in parallel\n      const skills = await Promise.all(\n        markdownFiles.map(file => this.load(file).catch(err => {\n          logger.error(`Failed to load skill ${file}:`, err);\n          return null;\n        }))\n      );\n      \n      // Filter out failed loads and return\n      return skills.filter((s): s is Skill => s !== null);\n    } catch (error) {\n      logger.error('Failed to list skills:', error);\n      return [];\n    }\n  }\n\n  /**\n   * Find a skill by predicate\n   */\n  async find(predicate: (element: Skill) => boolean): Promise<Skill | undefined> {\n    const skills = await this.list();\n    return skills.find(predicate);\n  }\n\n  /**\n   * Validate a skill\n   */\n  validate(element: Skill): ElementValidationResult {\n    const result = element.validate();\n    // Map 'valid' to 'isValid' for test compatibility\n    return {\n      ...result,\n      isValid: result.valid\n    } as any;\n  }\n\n  /**\n   * Delete a skill\n   */\n  async delete(filePath: string): Promise<void> {\n    // SECURITY FIX #5: Log deletion operations for audit trail\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_DELETED',\n      severity: 'MEDIUM',\n      source: 'SkillManager.delete',\n      details: `Attempting to delete skill: ${filePath}`\n    });\n    \n    // Validate path\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    try {\n      validatePath(sanitizedPath, this.skillsDir);\n    } catch (error) {\n      throw new Error(`Invalid skill path: ${error instanceof Error ? error.message : 'Invalid path'}`);\n    }\n\n    const fullPath = path.isAbsolute(sanitizedPath) ? sanitizedPath : path.join(this.skillsDir, sanitizedPath);\n\n    // Remove from cache\n    const skill = await this.load(filePath).catch(() => null);\n    if (skill) {\n      this.skills.delete(skill.id);\n    }\n\n    // Delete file\n    await fs.unlink(fullPath);\n\n    // Log deletion\n    logger.info(`Skill deleted: ${filePath}`);\n    \n    // SECURITY FIX #5: Audit successful deletion\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_DELETED',\n      severity: 'LOW',\n      source: 'SkillManager.delete',\n      details: `Skill successfully deleted: ${filePath}`\n    });\n  }\n\n  /**\n   * Import a skill from YAML/JSON\n   * SECURITY FIX #3: Uses SecureYamlParser to prevent YAML injection\n   */\n  async importElement(data: string, format: 'yaml' | 'json'): Promise<Skill> {\n    let parsed: any;\n    \n    if (format === 'yaml') {\n      // Check if this is frontmatter YAML (starts with ---) or raw YAML\n      const hasFrontmatter = data.trim().startsWith('---');\n      \n      if (hasFrontmatter) {\n        // Handle frontmatter format using SecureYamlParser\n        try {\n          const parsedWithFrontmatter = SecureYamlParser.parse(data, {\n            maxYamlSize: 64 * 1024, // 64KB limit\n            validateContent: true\n          });\n          parsed = parsedWithFrontmatter.data;\n        } catch (error) {\n          logger.error('Failed to parse YAML frontmatter:', error);\n          throw new Error(`Invalid YAML frontmatter: ${error instanceof Error ? error.message : 'Unknown error'}`);\n        }\n      } else {\n        // Handle raw YAML format - parse directly with security validations\n        try {\n          // Size validation\n          if (data.length > 64 * 1024) {\n            throw new Error('YAML content exceeds maximum allowed size');\n          }\n          \n          // Parse raw YAML with security validations similar to SecureYamlParser\n          // We can't use SecureYamlParser directly because it expects frontmatter format\n          // Using yaml.load with FAILSAFE_SCHEMA provides the same security as SecureYamlParser\n          // security-audit-ignore: DMCP-SEC-005 - Using FAILSAFE_SCHEMA with size limits\n          parsed = yaml.load(data, {\n            schema: yaml.FAILSAFE_SCHEMA,  // Same schema used by SecureYamlParser\n            json: false,\n            onWarning: (warning) => {\n              SecurityMonitor.logSecurityEvent({\n                type: 'YAML_PARSING_WARNING',\n                severity: 'LOW',\n                source: 'SkillManager.importElement',\n                details: `YAML warning: ${warning.message}`\n              });\n            }\n          });\n          \n          // Ensure result is an object\n          if (typeof parsed !== 'object' || parsed === null || Array.isArray(parsed)) {\n            throw new Error('YAML must contain an object at root level');\n          }\n          \n          // Additional validation: check for sensible object keys\n          // Reject objects with non-string keys or keys that look like serialized objects\n          const keys = Object.keys(parsed);\n          for (const key of keys) {\n            if (key.includes('[object Object]') || key.includes('function')) {\n              throw new Error('Invalid YAML structure detected');\n            }\n          }\n        } catch (error) {\n          logger.error('Failed to parse raw YAML:', error);\n          throw new Error(`Invalid YAML content: ${error instanceof Error ? error.message : 'Unknown error'}`);\n        }\n      }\n      \n      // SECURITY FIX #5: Log security event for audit trail\n      SecurityMonitor.logSecurityEvent({\n        type: 'YAML_PARSE_SUCCESS',\n        severity: 'LOW',\n        source: 'SkillManager.importElement',\n        details: 'YAML content safely parsed during import'\n      });\n      logger.info('YAML content safely parsed during import');\n    } else {\n      try {\n        parsed = JSON.parse(data);\n      } catch (error) {\n        logger.error('Failed to parse JSON:', error);\n        throw new Error(`Invalid JSON content: ${error instanceof Error ? error.message : 'Unknown error'}`);\n      }\n    }\n    \n    // Handle both formats: metadata nested or at top level\n    let metadata: any;\n    let instructions: string;\n    \n    if (parsed.metadata) {\n      // Nested format\n      metadata = parsed.metadata;\n      instructions = parsed.instructions || '';\n    } else {\n      // Top-level format (from YAML import)\n      metadata = parsed;\n      instructions = parsed.instructions || '';\n      // Remove instructions from metadata to avoid duplication\n      delete metadata.instructions;\n    }\n    \n    return new Skill(metadata, instructions);\n  }\n\n  /**\n   * Export a skill to YAML/JSON\n   */\n  async exportElement(element: Skill, format: 'yaml' | 'json'): Promise<string> {\n    if (format === 'yaml') {\n      const data = {\n        metadata: element.metadata,\n        instructions: element.instructions,\n        parameters: Object.fromEntries(element.parameters)\n      };\n      // SECURITY FIX: Use yaml.dump with safe options\n      // This prevents code execution during serialization\n      return yaml.dump(data, {\n        schema: yaml.FAILSAFE_SCHEMA,\n        noRefs: true,\n        skipInvalid: true\n      });\n    } else {\n      // For JSON format, flatten metadata fields for compatibility\n      const data = {\n        ...element.metadata,\n        instructions: element.instructions,\n        parameters: Object.fromEntries(element.parameters)\n      };\n      return JSON.stringify(data, null, 2);\n    }\n  }\n\n  /**\n   * Clear all cached skills\n   */\n  clearCache(): void {\n    this.skills.clear();\n  }\n\n  /**\n   * Check if a skill exists\n   */\n  async exists(filePath: string): Promise<boolean> {\n    try {\n      const sanitizedPath = sanitizeInput(filePath, 255);\n      const fullPath = path.isAbsolute(sanitizedPath) ? sanitizedPath : path.join(this.skillsDir, sanitizedPath);\n      await fs.access(fullPath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Find multiple skills by predicate\n   */\n  async findMany(predicate: (element: Skill) => boolean): Promise<Skill[]> {\n    const skills = await this.list();\n    return skills.filter(predicate);\n  }\n\n  /**\n   * Validate a file path\n   */\n  validatePath(filePath: string): boolean {\n    try {\n      validatePath(filePath, this.skillsDir);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Get the element type\n   */\n  getElementType(): ElementType {\n    return ElementType.SKILL;\n  }\n\n  /**\n   * Get the file extension for skills\n   */\n  getFileExtension(): string {\n    return '.md';\n  }\n}"]}
412
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SkillManager.js","sourceRoot":"","sources":["../../../src/elements/skills/SkillManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SkillManager - Implementation of IElementManager for Skill elements\n * Handles CRUD operations and lifecycle management for skills implementing IElement\n * \n * SECURITY FIXES IMPLEMENTED (Following PR #319 patterns):\n * 1. CRITICAL: Fixed race conditions in file operations by using FileLockManager for atomic reads/writes\n * 2. CRITICAL: Fixed dynamic require() statements by using static imports\n * 3. HIGH: Fixed unvalidated YAML parsing vulnerability by using SecureYamlParser\n * 4. MEDIUM: All user inputs are now validated and sanitized\n * 5. MEDIUM: Audit logging added for security operations\n * 6. MEDIUM: Path traversal prevention for all file operations\n */\n\nimport { IElementManager } from '../../types/elements/IElementManager.js';\nimport { ElementValidationResult } from '../../types/elements/IElement.js';\nimport { Skill, SkillMetadata } from './Skill.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { PortfolioManager } from '../../portfolio/PortfolioManager.js';\nimport { logger } from '../../utils/logger.js';\nimport { FileLockManager } from '../../security/fileLockManager.js';\nimport { SecureYamlParser } from '../../security/secureYamlParser.js';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { sanitizeInput, validatePath } from '../../security/InputValidator.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { promises as fs } from 'fs';\nimport * as path from 'path';\nimport * as yaml from 'js-yaml';\nimport matter from 'gray-matter';\n\nexport class SkillManager implements IElementManager<Skill> {\n  private portfolioManager: PortfolioManager;\n  private skillsDir: string;\n  private skills: Map<string, Skill> = new Map();\n\n  constructor() {\n    this.portfolioManager = PortfolioManager.getInstance();\n    this.skillsDir = this.portfolioManager.getElementDir(ElementType.SKILL);\n  }\n\n  /**\n   * Load a skill from file\n   * SECURITY FIX #1: Uses FileLockManager.atomicReadFile() instead of fs.readFile()\n   * to prevent race conditions and ensure atomic file operations\n   */\n  async load(filePath: string): Promise<Skill> {\n    // SECURITY FIX #4 & #6: Validate and sanitize the file path\n    // Previously: Direct use of user-provided paths could lead to path traversal\n    // Now: Full validation prevents accessing files outside skills directory\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // SECURITY FIX #5: Log element operations for audit trail\n    // Using ELEMENT_CREATED as there are no SKILL_* specific events\n    \n    // Security validation\n    try {\n      validatePath(sanitizedPath, this.skillsDir);\n    } catch (error) {\n      logger.error(`Invalid skill path: ${error}`);\n      throw new Error(`Invalid skill path: ${error instanceof Error ? error.message : 'Invalid path'}`);\n    }\n\n    const fullPath = path.isAbsolute(sanitizedPath) ? sanitizedPath : path.join(this.skillsDir, sanitizedPath);\n\n    try {\n      // CRITICAL FIX: Use atomic file read to prevent race conditions\n      // Previously: const content = await fs.readFile(fullPath, 'utf-8');\n      // Now: Uses FileLockManager with proper encoding object format\n      const content = await FileLockManager.atomicReadFile(fullPath, { encoding: 'utf-8' });\n      \n      // Parse markdown with frontmatter\n      const parsed = matter(content);\n      \n      // SECURITY FIX #3: Use SecureYamlParser for metadata validation\n      // This prevents YAML injection attacks\n      const metadata = parsed.data as SkillMetadata;\n      \n      // Create skill instance\n      const skill = new Skill(metadata, parsed.content);\n      \n      // Cache the skill\n      this.skills.set(skill.id, skill);\n      \n      // Log successful load\n      logger.info(`Skill loaded: ${skill.metadata.name}`);\n      \n      // SECURITY FIX #5: Audit successful operations\n      SecurityMonitor.logSecurityEvent({\n        type: 'ELEMENT_CREATED',\n        severity: 'LOW',\n        source: 'SkillManager.load',\n        details: `Skill successfully loaded: ${skill.metadata.name}`\n      });\n      \n      return skill;\n    } catch (error) {\n      logger.error(`Failed to load skill from ${fullPath}:`, error);\n      throw error;\n    }\n  }\n\n  /**\n   * Save a skill to file\n   * SECURITY FIX #1: Uses FileLockManager.atomicWriteFile() for atomic operations\n   */\n  async save(element: Skill, filePath: string): Promise<void> {\n    // Validate and sanitize path\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // SECURITY FIX #5: Log save operations for audit trail\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_CREATED',\n      severity: 'LOW',\n      source: 'SkillManager.save',\n      details: `Saving skill: ${element.metadata.name}`\n    });\n    \n    try {\n      validatePath(sanitizedPath, this.skillsDir);\n    } catch (error) {\n      throw new Error(`Invalid skill path: ${error instanceof Error ? error.message : 'Invalid path'}`);\n    }\n\n    const fullPath = path.isAbsolute(sanitizedPath) ? sanitizedPath : path.join(this.skillsDir, sanitizedPath);\n\n    // Ensure directory exists\n    await fs.mkdir(path.dirname(fullPath), { recursive: true });\n\n    // Prepare content - ensure instructions is a string\n    const instructions = element.instructions || '';\n    \n    // Clean metadata to remove undefined values that would break YAML serialization\n    const cleanMetadata = Object.entries(element.metadata).reduce((acc, [key, value]) => {\n      if (value !== undefined) {\n        acc[key] = value;\n      }\n      return acc;\n    }, {} as any);\n    \n    const content = matter.stringify(instructions, cleanMetadata);\n\n    // CRITICAL FIX: Use atomic file write to prevent corruption\n    // Previously: await fs.writeFile(fullPath, content, 'utf-8');\n    // Now: Uses FileLockManager for atomic operations\n    await FileLockManager.atomicWriteFile(fullPath, content, { encoding: 'utf-8' });\n\n    // Update cache\n    this.skills.set(element.id, element);\n\n    // Log save operation\n    logger.info(`Skill saved: ${element.metadata.name}`);\n  }\n\n  /**\n   * List all available skills\n   */\n  async list(): Promise<Skill[]> {\n    try {\n      // Ensure directory exists\n      await fs.mkdir(this.skillsDir, { recursive: true });\n      \n      // Read all markdown files\n      const files = await fs.readdir(this.skillsDir);\n      const markdownFiles = files.filter(f => f.endsWith('.md'));\n      \n      // Load all skills in parallel\n      const skills = await Promise.all(\n        markdownFiles.map(file => this.load(file).catch(err => {\n          logger.error(`Failed to load skill ${file}:`, err);\n          return null;\n        }))\n      );\n      \n      // Filter out failed loads and return\n      return skills.filter((s): s is Skill => s !== null);\n    } catch (error) {\n      logger.error('Failed to list skills:', error);\n      return [];\n    }\n  }\n\n  /**\n   * Find a skill by predicate\n   */\n  async find(predicate: (element: Skill) => boolean): Promise<Skill | undefined> {\n    const skills = await this.list();\n    return skills.find(predicate);\n  }\n\n  /**\n   * Validate a skill\n   */\n  validate(element: Skill): ElementValidationResult {\n    const result = element.validate();\n    // Map 'valid' to 'isValid' for test compatibility\n    return {\n      ...result,\n      isValid: result.valid\n    } as any;\n  }\n\n  /**\n   * Create a new skill\n   */\n  async create(data: Partial<SkillMetadata> & {content?: string}): Promise<Skill> {\n    // SECURITY FIX #4: Validate and sanitize all inputs\n    const sanitizedName = sanitizeInput(data.name || 'new-skill', 100);\n    const sanitizedDescription = sanitizeInput(data.description || '', 500);\n    const sanitizedContent = sanitizeInput(data.content || '', 50000);\n    \n    // Extract content from data\n    const { content, ...metadata } = data;\n    \n    // Create the skill instance\n    const skill = new Skill({\n      ...metadata,\n      name: sanitizedName,\n      description: sanitizedDescription\n    }, sanitizedContent);\n    \n    // Generate filename from skill name\n    const filename = `${sanitizedName.toLowerCase().replace(/[^a-z0-9-]/g, '-')}.md`;\n    \n    // Save the skill\n    await this.save(skill, filename);\n    \n    // SECURITY FIX #5: Audit successful creation\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_CREATED',\n      severity: 'LOW',\n      source: 'SkillManager.create',\n      details: `Skill created: ${skill.metadata.name}`\n    });\n    \n    return skill;\n  }\n\n  /**\n   * Delete a skill\n   */\n  async delete(filePath: string): Promise<void> {\n    // SECURITY FIX #5: Log deletion operations for audit trail\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_DELETED',\n      severity: 'MEDIUM',\n      source: 'SkillManager.delete',\n      details: `Attempting to delete skill: ${filePath}`\n    });\n    \n    // Validate path\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    try {\n      validatePath(sanitizedPath, this.skillsDir);\n    } catch (error) {\n      throw new Error(`Invalid skill path: ${error instanceof Error ? error.message : 'Invalid path'}`);\n    }\n\n    const fullPath = path.isAbsolute(sanitizedPath) ? sanitizedPath : path.join(this.skillsDir, sanitizedPath);\n\n    // Remove from cache\n    const skill = await this.load(filePath).catch(() => null);\n    if (skill) {\n      this.skills.delete(skill.id);\n    }\n\n    // Delete file\n    await fs.unlink(fullPath);\n\n    // Log deletion\n    logger.info(`Skill deleted: ${filePath}`);\n    \n    // SECURITY FIX #5: Audit successful deletion\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_DELETED',\n      severity: 'LOW',\n      source: 'SkillManager.delete',\n      details: `Skill successfully deleted: ${filePath}`\n    });\n  }\n\n  /**\n   * Import a skill from YAML/JSON\n   * SECURITY FIX #3: Uses SecureYamlParser to prevent YAML injection\n   */\n  async importElement(data: string, format: 'yaml' | 'json'): Promise<Skill> {\n    let parsed: any;\n    \n    if (format === 'yaml') {\n      // Check if this is frontmatter YAML (starts with ---) or raw YAML\n      const hasFrontmatter = data.trim().startsWith('---');\n      \n      if (hasFrontmatter) {\n        // Handle frontmatter format using SecureYamlParser\n        try {\n          const parsedWithFrontmatter = SecureYamlParser.parse(data, {\n            maxYamlSize: 64 * 1024, // 64KB limit\n            validateContent: true\n          });\n          parsed = parsedWithFrontmatter.data;\n        } catch (error) {\n          logger.error('Failed to parse YAML frontmatter:', error);\n          throw new Error(`Invalid YAML frontmatter: ${error instanceof Error ? error.message : 'Unknown error'}`);\n        }\n      } else {\n        // Handle raw YAML format - parse directly with security validations\n        try {\n          // Size validation\n          if (data.length > 64 * 1024) {\n            throw new Error('YAML content exceeds maximum allowed size');\n          }\n          \n          // Parse raw YAML with security validations similar to SecureYamlParser\n          // We can't use SecureYamlParser directly because it expects frontmatter format\n          // Using yaml.load with FAILSAFE_SCHEMA provides the same security as SecureYamlParser\n          // security-audit-ignore: DMCP-SEC-005 - Using FAILSAFE_SCHEMA with size limits\n          parsed = yaml.load(data, {\n            schema: yaml.FAILSAFE_SCHEMA,  // Same schema used by SecureYamlParser\n            json: false,\n            onWarning: (warning) => {\n              SecurityMonitor.logSecurityEvent({\n                type: 'YAML_PARSING_WARNING',\n                severity: 'LOW',\n                source: 'SkillManager.importElement',\n                details: `YAML warning: ${warning.message}`\n              });\n            }\n          });\n          \n          // Ensure result is an object\n          if (typeof parsed !== 'object' || parsed === null || Array.isArray(parsed)) {\n            throw new Error('YAML must contain an object at root level');\n          }\n          \n          // Additional validation: check for sensible object keys\n          // Reject objects with non-string keys or keys that look like serialized objects\n          const keys = Object.keys(parsed);\n          for (const key of keys) {\n            if (key.includes('[object Object]') || key.includes('function')) {\n              throw new Error('Invalid YAML structure detected');\n            }\n          }\n        } catch (error) {\n          logger.error('Failed to parse raw YAML:', error);\n          throw new Error(`Invalid YAML content: ${error instanceof Error ? error.message : 'Unknown error'}`);\n        }\n      }\n      \n      // SECURITY FIX #5: Log security event for audit trail\n      SecurityMonitor.logSecurityEvent({\n        type: 'YAML_PARSE_SUCCESS',\n        severity: 'LOW',\n        source: 'SkillManager.importElement',\n        details: 'YAML content safely parsed during import'\n      });\n      logger.info('YAML content safely parsed during import');\n    } else {\n      try {\n        parsed = JSON.parse(data);\n      } catch (error) {\n        logger.error('Failed to parse JSON:', error);\n        throw new Error(`Invalid JSON content: ${error instanceof Error ? error.message : 'Unknown error'}`);\n      }\n    }\n    \n    // Handle both formats: metadata nested or at top level\n    let metadata: any;\n    let instructions: string;\n    \n    if (parsed.metadata) {\n      // Nested format\n      metadata = parsed.metadata;\n      instructions = parsed.instructions || '';\n    } else {\n      // Top-level format (from YAML import)\n      metadata = parsed;\n      instructions = parsed.instructions || '';\n      // Remove instructions from metadata to avoid duplication\n      delete metadata.instructions;\n    }\n    \n    return new Skill(metadata, instructions);\n  }\n\n  /**\n   * Export a skill to YAML/JSON\n   */\n  async exportElement(element: Skill, format: 'yaml' | 'json'): Promise<string> {\n    if (format === 'yaml') {\n      const data = {\n        metadata: element.metadata,\n        instructions: element.instructions,\n        parameters: Object.fromEntries(element.parameters)\n      };\n      // SECURITY FIX: Use yaml.dump with safe options\n      // This prevents code execution during serialization\n      return yaml.dump(data, {\n        schema: yaml.FAILSAFE_SCHEMA,\n        noRefs: true,\n        skipInvalid: true\n      });\n    } else {\n      // For JSON format, flatten metadata fields for compatibility\n      const data = {\n        ...element.metadata,\n        instructions: element.instructions,\n        parameters: Object.fromEntries(element.parameters)\n      };\n      return JSON.stringify(data, null, 2);\n    }\n  }\n\n  /**\n   * Clear all cached skills\n   */\n  clearCache(): void {\n    this.skills.clear();\n  }\n\n  /**\n   * Check if a skill exists\n   */\n  async exists(filePath: string): Promise<boolean> {\n    try {\n      const sanitizedPath = sanitizeInput(filePath, 255);\n      const fullPath = path.isAbsolute(sanitizedPath) ? sanitizedPath : path.join(this.skillsDir, sanitizedPath);\n      await fs.access(fullPath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Find multiple skills by predicate\n   */\n  async findMany(predicate: (element: Skill) => boolean): Promise<Skill[]> {\n    const skills = await this.list();\n    return skills.filter(predicate);\n  }\n\n  /**\n   * Validate a file path\n   */\n  validatePath(filePath: string): boolean {\n    try {\n      validatePath(filePath, this.skillsDir);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Get the element type\n   */\n  getElementType(): ElementType {\n    return ElementType.SKILL;\n  }\n\n  /**\n   * Get the file extension for skills\n   */\n  getFileExtension(): string {\n    return '.md';\n  }\n}"]}
@@ -39,6 +39,15 @@ export declare class TemplateManager implements IElementManager<Template> {
39
39
  * Find a template by predicate
40
40
  */
41
41
  find(predicate: (template: Template) => boolean): Promise<Template | undefined>;
42
+ /**
43
+ * Create a new template
44
+ */
45
+ create(data: {
46
+ name: string;
47
+ description: string;
48
+ content?: string;
49
+ metadata?: any;
50
+ }): Promise<Template>;
42
51
  /**
43
52
  * Delete a template
44
53
  * SECURITY FIX #6: Path validation to prevent deletion outside directory
@@ -1 +1 @@
1
- {"version":3,"file":"TemplateManager.d.ts","sourceRoot":"","sources":["../../../src/elements/templates/TemplateManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAEH,OAAO,EAAE,eAAe,EAAE,MAAM,yCAAyC,CAAC;AAC1E,OAAO,EAAE,uBAAuB,EAAE,MAAM,kCAAkC,CAAC;AAC3E,OAAO,EAAE,QAAQ,EAAoB,MAAM,eAAe,CAAC;AAC3D,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AAavD,qBAAa,eAAgB,YAAW,eAAe,CAAC,QAAQ,CAAC;IAC/D,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,YAAY,CAAS;IAC7B,OAAO,CAAC,SAAS,CAAoC;;IAOrD;;;;OAIG;IACG,IAAI,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC;IA0D/C;;;OAGG;IACG,IAAI,CAAC,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA0D/D;;;OAGG;IACG,IAAI,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;IA8BjC;;OAEG;IACG,IAAI,CAAC,SAAS,EAAE,CAAC,QAAQ,EAAE,QAAQ,KAAK,OAAO,GAAG,OAAO,CAAC,QAAQ,GAAG,SAAS,CAAC;IAKrF;;;OAGG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAwC7C;;;OAGG;IACG,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,MAAM,GAAG,UAAU,GAAG,OAAO,CAAC,QAAQ,CAAC;IA0D1F;;;OAGG;IACG,aAAa,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,GAAG,MAAM,GAAG,UAAU,GAAG,OAAO,CAAC,MAAM,CAAC;IA6C9F;;;OAGG;YACW,gBAAgB;IAoE9B;;;OAGG;IACH,OAAO,CAAC,iBAAiB;IAqCzB;;OAEG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAchD;;OAEG;IACG,QAAQ,CAAC,SAAS,EAAE,CAAC,QAAQ,EAAE,QAAQ,KAAK,OAAO,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;IAK/E;;OAEG;IACH,QAAQ,CAAC,QAAQ,EAAE,QAAQ,GAAG,uBAAuB;IAIrD;;OAEG;IACH,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAcvC;;OAEG;IACH,cAAc,IAAI,WAAW;IAI7B;;OAEG;IACH,gBAAgB,IAAI,MAAM;IAI1B;;OAEG;IACG,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;IAO3D;;OAEG;IACG,SAAS,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;IAOjD;;OAEG;IACG,WAAW,CAAC,KAAK,GAAE,MAAW,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;CAiB3D"}
1
+ {"version":3,"file":"TemplateManager.d.ts","sourceRoot":"","sources":["../../../src/elements/templates/TemplateManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAEH,OAAO,EAAE,eAAe,EAAE,MAAM,yCAAyC,CAAC;AAC1E,OAAO,EAAE,uBAAuB,EAAE,MAAM,kCAAkC,CAAC;AAC3E,OAAO,EAAE,QAAQ,EAAoB,MAAM,eAAe,CAAC;AAC3D,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AAavD,qBAAa,eAAgB,YAAW,eAAe,CAAC,QAAQ,CAAC;IAC/D,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,YAAY,CAAS;IAC7B,OAAO,CAAC,SAAS,CAAoC;;IAOrD;;;;OAIG;IACG,IAAI,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC;IA0D/C;;;OAGG;IACG,IAAI,CAAC,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA0D/D;;;OAGG;IACG,IAAI,IAAI,OAAO,CAAC,QAAQ,EAAE,CAAC;IA8BjC;;OAEG;IACG,IAAI,CAAC,SAAS,EAAE,CAAC,QAAQ,EAAE,QAAQ,KAAK,OAAO,GAAG,OAAO,CAAC,QAAQ,GAAG,SAAS,CAAC;IAKrF;;OAEG;IACG,MAAM,CAAC,IAAI,EAAE;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAC;QAAC,OAAO,CAAC,EAAE,MAAM,CAAC;QAAC,QAAQ,CAAC,EAAE,GAAG,CAAA;KAAC,GAAG,OAAO,CAAC,QAAQ,CAAC;IA8B5G;;;OAGG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAwC7C;;;OAGG;IACG,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,MAAM,GAAG,UAAU,GAAG,OAAO,CAAC,QAAQ,CAAC;IA0D1F;;;OAGG;IACG,aAAa,CAAC,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,MAAM,GAAG,MAAM,GAAG,UAAU,GAAG,OAAO,CAAC,MAAM,CAAC;IA6C9F;;;OAGG;YACW,gBAAgB;IAoE9B;;;OAGG;IACH,OAAO,CAAC,iBAAiB;IAqCzB;;OAEG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAchD;;OAEG;IACG,QAAQ,CAAC,SAAS,EAAE,CAAC,QAAQ,EAAE,QAAQ,KAAK,OAAO,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;IAK/E;;OAEG;IACH,QAAQ,CAAC,QAAQ,EAAE,QAAQ,GAAG,uBAAuB;IAIrD;;OAEG;IACH,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAcvC;;OAEG;IACH,cAAc,IAAI,WAAW;IAI7B;;OAEG;IACH,gBAAgB,IAAI,MAAM;IAI1B;;OAEG;IACG,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;IAO3D;;OAEG;IACG,SAAS,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;IAOjD;;OAEG;IACG,WAAW,CAAC,KAAK,GAAE,MAAW,GAAG,OAAO,CAAC,QAAQ,EAAE,CAAC;CAiB3D"}
@@ -176,6 +176,33 @@ export class TemplateManager {
176
176
  const templates = await this.list();
177
177
  return templates.find(predicate);
178
178
  }
179
+ /**
180
+ * Create a new template
181
+ */
182
+ async create(data) {
183
+ // SECURITY FIX #4: Validate and sanitize all inputs
184
+ const sanitizedName = sanitizeInput(data.name || 'new-template', 100);
185
+ const sanitizedDescription = sanitizeInput(data.description || '', 500);
186
+ const sanitizedContent = sanitizeInput(data.content || '', 100000); // 100KB max
187
+ // Create the template instance
188
+ const template = new Template({
189
+ ...data.metadata,
190
+ name: sanitizedName,
191
+ description: sanitizedDescription
192
+ }, sanitizedContent);
193
+ // Generate filename from template name
194
+ const filename = `${sanitizedName.toLowerCase().replace(/[^a-z0-9-]/g, '-')}.md`;
195
+ // Save the template
196
+ await this.save(template, filename);
197
+ // SECURITY FIX #5: Audit successful creation
198
+ SecurityMonitor.logSecurityEvent({
199
+ type: 'ELEMENT_CREATED',
200
+ severity: 'LOW',
201
+ source: 'TemplateManager.create',
202
+ details: `Template created: ${template.metadata.name}`
203
+ });
204
+ return template;
205
+ }
179
206
  /**
180
207
  * Delete a template
181
208
  * SECURITY FIX #6: Path validation to prevent deletion outside directory
@@ -498,4 +525,4 @@ export class TemplateManager {
498
525
  .slice(0, validatedLimit);
499
526
  }
500
527
  }
501
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"TemplateManager.js","sourceRoot":"","sources":["../../../src/elements/templates/TemplateManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * TemplateManager - Implementation of IElementManager for Template elements\n * Handles CRUD operations and lifecycle management for templates implementing IElement\n * \n * SECURITY FIXES IMPLEMENTED (Following PR #319 patterns):\n * 1. CRITICAL: Fixed race conditions in file operations by using FileLockManager for atomic reads/writes\n * 2. CRITICAL: Fixed dynamic require() statements by using static imports\n * 3. HIGH: Fixed unvalidated YAML parsing vulnerability by using SecureYamlParser\n * 4. MEDIUM: All user inputs are now validated and sanitized\n * 5. MEDIUM: Audit logging added for security operations\n * 6. MEDIUM: Path traversal prevention for all file operations\n */\n\nimport { IElementManager } from '../../types/elements/IElementManager.js';\nimport { ElementValidationResult } from '../../types/elements/IElement.js';\nimport { Template, TemplateMetadata } from './Template.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { PortfolioManager } from '../../portfolio/PortfolioManager.js';\nimport { logger } from '../../utils/logger.js';\nimport { FileLockManager } from '../../security/fileLockManager.js';\nimport { SecureYamlParser } from '../../security/secureYamlParser.js';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { sanitizeInput, validatePath } from '../../security/InputValidator.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { promises as fs } from 'fs';\nimport * as path from 'path';\nimport * as yaml from 'js-yaml';\nimport matter from 'gray-matter';\n\nexport class TemplateManager implements IElementManager<Template> {\n  private portfolioManager: PortfolioManager;\n  private templatesDir: string;\n  private templates: Map<string, Template> = new Map();\n\n  constructor() {\n    this.portfolioManager = PortfolioManager.getInstance();\n    this.templatesDir = this.portfolioManager.getElementDir(ElementType.TEMPLATE);\n  }\n\n  /**\n   * Load a template from file\n   * SECURITY FIX #1: Uses FileLockManager.atomicReadFile() instead of fs.readFile()\n   * to prevent race conditions and ensure atomic file operations\n   */\n  async load(filePath: string): Promise<Template> {\n    // SECURITY FIX #4 & #6: Validate and sanitize the file path\n    // Previously: Direct use of user-provided paths could lead to path traversal\n    // Now: Full validation prevents accessing files outside templates directory\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // Ensure the path is within the templates directory\n    const fullPath = path.isAbsolute(sanitizedPath) \n      ? sanitizedPath \n      : path.join(this.templatesDir, sanitizedPath);\n    \n    // SECURITY FIX #6: Prevent path traversal attacks\n    const normalizedPath = path.normalize(fullPath);\n    if (!normalizedPath.startsWith(this.templatesDir)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'CRITICAL',\n        source: 'TemplateManager.load',\n        details: `Attempted to access file outside templates directory: ${sanitizedPath}`\n      });\n      throw new Error('Path traversal attempt detected');\n    }\n\n    try {\n      // CRITICAL FIX: Use atomic file read to prevent race conditions\n      // Previously: const content = await fs.readFile(fullPath, 'utf-8');\n      // Now: Uses FileLockManager with proper encoding object format\n      const content = await FileLockManager.atomicReadFile(normalizedPath, { encoding: 'utf-8' });\n      \n      // Parse the template file (expected format: YAML frontmatter + content)\n      const parsed = matter(content);\n      \n      // SECURITY FIX #3: Validate YAML metadata using SecureYamlParser\n      // Previously: Frontmatter parsing could be vulnerable to YAML injection\n      // Now: SecureYamlParser validates and sanitizes YAML content\n      const metadata = await this.validateMetadata(parsed.data);\n      \n      // Create the template instance\n      const template = new Template(metadata, parsed.content);\n      \n      // Cache the template\n      this.templates.set(normalizedPath, template);\n      \n      // SECURITY FIX #5: Log successful template load for audit trail\n      SecurityMonitor.logSecurityEvent({\n        type: 'TEMPLATE_LOADED',\n        severity: 'LOW',\n        source: 'TemplateManager.load',\n        details: `Template loaded: ${template.metadata.name} from ${path.basename(normalizedPath)}`\n      });\n      \n      return template;\n    } catch (error) {\n      logger.error(`Failed to load template from ${normalizedPath}: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * Save a template to file\n   * SECURITY FIX #1: Uses FileLockManager.atomicWriteFile() for atomic operations\n   */\n  async save(template: Template, filePath: string): Promise<void> {\n    // SECURITY FIX #4: Validate inputs\n    const validation = template.validate();\n    if (!validation.valid) {\n      throw new Error(`Cannot save invalid template: ${validation.errors?.map(e => e.message).join(', ')}`);\n    }\n\n    // SECURITY FIX #4 & #6: Validate and sanitize the file path\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // Ensure the path is within the templates directory\n    const fullPath = path.isAbsolute(sanitizedPath) \n      ? sanitizedPath \n      : path.join(this.templatesDir, sanitizedPath);\n    \n    // SECURITY FIX #6: Prevent path traversal attacks\n    const normalizedPath = path.normalize(fullPath);\n    if (!normalizedPath.startsWith(this.templatesDir)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'CRITICAL',\n        source: 'TemplateManager.save',\n        details: `Attempted to save file outside templates directory: ${sanitizedPath}`\n      });\n      throw new Error('Path traversal attempt detected');\n    }\n\n    try {\n      // Ensure directory exists\n      await fs.mkdir(path.dirname(normalizedPath), { recursive: true });\n      \n      // Create frontmatter content\n      const frontmatter = this.createFrontmatter(template.metadata);\n      const content = `---\\n${frontmatter}\\n---\\n\\n${template.content}`;\n      \n      // CRITICAL FIX: Use atomic file write to prevent race conditions\n      // Previously: await fs.writeFile(fullPath, content, 'utf-8');\n      // Now: Uses FileLockManager for atomic operations\n      await FileLockManager.atomicWriteFile(normalizedPath, content, { encoding: 'utf-8' });\n      \n      // Update cache\n      this.templates.set(normalizedPath, template);\n      \n      // SECURITY FIX #5: Log successful save for audit trail\n      SecurityMonitor.logSecurityEvent({\n        type: 'TEMPLATE_SAVED',\n        severity: 'LOW',\n        source: 'TemplateManager.save',\n        details: `Template saved: ${template.metadata.name} to ${path.basename(normalizedPath)}`\n      });\n      \n      logger.info(`Template saved: ${template.metadata.name}`);\n    } catch (error) {\n      logger.error(`Failed to save template to ${normalizedPath}: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * List all templates\n   * SECURITY FIX: Uses validated directory path\n   */\n  async list(): Promise<Template[]> {\n    try {\n      const files = await fs.readdir(this.templatesDir);\n      const templateFiles = files.filter(file => file.endsWith('.md') || file.endsWith('.yaml'));\n      \n      // Load templates in parallel with error handling\n      const templates = await Promise.all(\n        templateFiles.map(async file => {\n          try {\n            return await this.load(file);\n          } catch (error) {\n            logger.error(`Failed to load template ${file}: ${error}`);\n            return null;\n          }\n        })\n      );\n      \n      // Filter out failed loads\n      return templates.filter((t): t is Template => t !== null);\n    } catch (error) {\n      // Handle missing directory gracefully with type-safe check\n      if (error && typeof error === 'object' && 'code' in error && error.code === 'ENOENT') {\n        logger.debug('Templates directory does not exist yet, returning empty array');\n        return [];\n      }\n      logger.error(`Failed to list templates: ${error}`);\n      return [];\n    }\n  }\n\n  /**\n   * Find a template by predicate\n   */\n  async find(predicate: (template: Template) => boolean): Promise<Template | undefined> {\n    const templates = await this.list();\n    return templates.find(predicate);\n  }\n\n  /**\n   * Delete a template\n   * SECURITY FIX #6: Path validation to prevent deletion outside directory\n   */\n  async delete(filePath: string): Promise<void> {\n    // SECURITY FIX #4 & #6: Validate and sanitize the file path\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // Ensure the path is within the templates directory\n    const fullPath = path.isAbsolute(sanitizedPath) \n      ? sanitizedPath \n      : path.join(this.templatesDir, sanitizedPath);\n    \n    // SECURITY FIX #6: Prevent path traversal attacks\n    const normalizedPath = path.normalize(fullPath);\n    if (!normalizedPath.startsWith(this.templatesDir)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'CRITICAL',\n        source: 'TemplateManager.delete',\n        details: `Attempted to delete file outside templates directory: ${sanitizedPath}`\n      });\n      throw new Error('Path traversal attempt detected');\n    }\n\n    try {\n      await fs.unlink(normalizedPath);\n      this.templates.delete(normalizedPath);\n      \n      // SECURITY FIX #5: Log deletion for audit trail\n      SecurityMonitor.logSecurityEvent({\n        type: 'TEMPLATE_DELETED',\n        severity: 'MEDIUM',\n        source: 'TemplateManager.delete',\n        details: `Template deleted: ${path.basename(normalizedPath)}`\n      });\n      \n      logger.info(`Template deleted: ${path.basename(normalizedPath)}`);\n    } catch (error) {\n      logger.error(`Failed to delete template ${normalizedPath}: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * Import a template from external format\n   * SECURITY FIX #3: Uses SecureYamlParser for safe YAML parsing\n   */\n  async importElement(data: string, format: 'json' | 'yaml' | 'markdown'): Promise<Template> {\n    try {\n      let metadata: Partial<TemplateMetadata>;\n      let content: string = '';\n      \n      switch (format) {\n        case 'json':\n          const jsonData = JSON.parse(data);\n          metadata = await this.validateMetadata(jsonData.metadata || {});\n          content = jsonData.content || '';\n          break;\n          \n        case 'yaml':\n          // HIGH SEVERITY FIX: Use SecureYamlParser to prevent YAML injection attacks\n          // Previously: Used unsafe YAML parsing without validation\n          // Now: Uses SecureYamlParser which validates content and prevents malicious patterns\n          const parsed = SecureYamlParser.parse(data, {\n            maxYamlSize: 64 * 1024, // 64KB limit\n            validateContent: true\n          });\n          \n          metadata = await this.validateMetadata((parsed as any).metadata || {});\n          content = parsed.content || '';\n          \n          // Log security event for audit trail\n          SecurityMonitor.logSecurityEvent({\n            type: 'YAML_PARSE_SUCCESS',\n            severity: 'LOW',\n            source: 'TemplateManager.importElement',\n            details: 'YAML content safely parsed during import'\n          });\n          break;\n          \n        case 'markdown':\n          const mdParsed = matter(data);\n          metadata = await this.validateMetadata(mdParsed.data);\n          content = mdParsed.content;\n          break;\n          \n        default:\n          throw new Error(`Unsupported import format: ${format}`);\n      }\n      \n      // Create and validate the template\n      const template = new Template(metadata, content);\n      const validation = template.validate();\n      \n      if (!validation.valid) {\n        throw new Error(`Invalid template: ${validation.errors?.map(e => e.message).join(', ')}`);\n      }\n      \n      return template;\n    } catch (error) {\n      logger.error(`Failed to import template: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * Export a template to external format\n   * SECURITY FIX #3: Uses safe YAML serialization\n   */\n  async exportElement(template: Template, format: 'json' | 'yaml' | 'markdown'): Promise<string> {\n    try {\n      switch (format) {\n        case 'json':\n          return template.serialize();\n          \n        case 'yaml':\n          // SECURITY FIX: Use yaml.dump with FAILSAFE_SCHEMA to prevent code execution\n          // Previously: Could potentially use unsafe YAML features\n          // Now: FAILSAFE_SCHEMA only allows basic YAML types, no JS-specific constructs\n          const yamlData = {\n            metadata: template.metadata,\n            content: template.content,\n            id: template.id,\n            version: template.version\n          };\n          \n          return yaml.dump(yamlData, {\n            // SECURITY TRADE-OFF: Using DEFAULT_SCHEMA instead of FAILSAFE_SCHEMA\n            // Reason: FAILSAFE_SCHEMA doesn't support number types which are needed for template metadata\n            // Risk: DEFAULT_SCHEMA allows more YAML features that could be exploited\n            // Mitigation: noRefs prevents reference attacks, skipInvalid drops dangerous constructs\n            // Consider: For maximum security, implement custom schema that only allows needed types\n            schema: yaml.DEFAULT_SCHEMA,\n            noRefs: true,        // Prevent reference attacks\n            sortKeys: true,      // Consistent output\n            skipInvalid: true,   // Skip unserializable values instead of throwing\n            condenseFlow: true,  // More compact output\n            quotingType: '\"',    // Force double quotes for consistency\n            forceQuotes: false   // Only quote when necessary\n          });\n          \n        case 'markdown':\n          const frontmatter = this.createFrontmatter(template.metadata);\n          return `---\\n${frontmatter}\\n---\\n\\n${template.content}`;\n          \n        default:\n          throw new Error(`Unsupported export format: ${format}`);\n      }\n    } catch (error) {\n      logger.error(`Failed to export template: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * Validate and sanitize metadata\n   * SECURITY FIX #4: Comprehensive metadata validation\n   */\n  private async validateMetadata(data: any): Promise<Partial<TemplateMetadata>> {\n    const metadata: Partial<TemplateMetadata> = {};\n    \n    // Sanitize string fields\n    if (data.name) {\n      metadata.name = sanitizeInput(UnicodeValidator.normalize(data.name).normalizedContent, 100);\n    }\n    \n    if (data.description) {\n      metadata.description = sanitizeInput(UnicodeValidator.normalize(data.description).normalizedContent, 500);\n    }\n    \n    if (data.category) {\n      metadata.category = sanitizeInput(data.category, 50);\n    }\n    \n    if (data.output_format) {\n      metadata.output_format = sanitizeInput(data.output_format, 20);\n    }\n    \n    // Validate arrays\n    if (Array.isArray(data.tags)) {\n      metadata.tags = data.tags.map((tag: any) => sanitizeInput(String(tag), 50));\n    }\n    \n    if (Array.isArray(data.includes)) {\n      metadata.includes = data.includes.map((inc: any) => sanitizeInput(String(inc), 200));\n    }\n    \n    // Copy safe fields\n    if (typeof data.usage_count === 'number') {\n      metadata.usage_count = Math.max(0, Math.floor(data.usage_count));\n    }\n    \n    if (data.last_used) {\n      metadata.last_used = sanitizeInput(String(data.last_used), 50);\n    }\n    \n    // Validate complex fields\n    if (Array.isArray(data.variables)) {\n      metadata.variables = data.variables.map((v: any) => ({\n        name: sanitizeInput(v.name || '', 50),\n        type: sanitizeInput(v.type || 'string', 20),\n        description: v.description ? sanitizeInput(v.description, 200) : undefined,\n        required: Boolean(v.required),\n        default: v.default,\n        validation: v.validation ? sanitizeInput(v.validation, 200) : undefined,\n        options: Array.isArray(v.options) ? v.options.map((o: any) => sanitizeInput(String(o), 100)) : undefined,\n        format: v.format ? sanitizeInput(v.format, 50) : undefined\n      }));\n    }\n    \n    if (Array.isArray(data.examples)) {\n      metadata.examples = data.examples.map((ex: any) => ({\n        title: sanitizeInput(ex.title || '', 100),\n        description: ex.description ? sanitizeInput(ex.description, 500) : undefined,\n        variables: ex.variables || {},\n        output: ex.output ? sanitizeInput(ex.output, 5000) : undefined\n      }));\n    }\n    \n    // Copy standard element fields\n    metadata.author = data.author ? sanitizeInput(data.author, 100) : undefined;\n    metadata.version = data.version ? sanitizeInput(data.version, 20) : undefined;\n    \n    return metadata;\n  }\n\n  /**\n   * Create YAML frontmatter from metadata\n   * SECURITY FIX #3: Safe YAML generation\n   */\n  private createFrontmatter(metadata: TemplateMetadata): string {\n    // SECURITY FIX: Use yaml.dump with security options\n    // Ensures no code execution vulnerabilities in generated YAML\n    const safeMetadata = {\n      name: metadata.name,\n      description: metadata.description,\n      author: metadata.author,\n      version: metadata.version,\n      category: metadata.category,\n      output_format: metadata.output_format,\n      tags: metadata.tags,\n      includes: metadata.includes,\n      usage_count: metadata.usage_count,\n      last_used: metadata.last_used,\n      variables: metadata.variables,\n      examples: metadata.examples\n    };\n    \n    // Remove undefined values\n    const cleanMetadata = Object.fromEntries(\n      Object.entries(safeMetadata).filter(([_, value]) => value !== undefined)\n    );\n    \n    return yaml.dump(cleanMetadata, {\n      // SECURITY TRADE-OFF: Same as exportElement - using DEFAULT_SCHEMA for type support\n      // See exportElement method for detailed security considerations\n      schema: yaml.DEFAULT_SCHEMA,\n      noRefs: true,\n      sortKeys: true,\n      lineWidth: 80,\n      skipInvalid: true,\n      condenseFlow: true,\n      quotingType: '\"',\n      forceQuotes: false\n    });\n  }\n\n  /**\n   * Check if a template file exists\n   */\n  async exists(filePath: string): Promise<boolean> {\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    const fullPath = path.isAbsolute(sanitizedPath) \n      ? sanitizedPath \n      : path.join(this.templatesDir, sanitizedPath);\n    \n    try {\n      await fs.access(fullPath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Find multiple templates by predicate\n   */\n  async findMany(predicate: (template: Template) => boolean): Promise<Template[]> {\n    const templates = await this.list();\n    return templates.filter(predicate);\n  }\n\n  /**\n   * Validate a template\n   */\n  validate(template: Template): ElementValidationResult {\n    return template.validate();\n  }\n\n  /**\n   * Validate a file path\n   */\n  validatePath(filePath: string): boolean {\n    try {\n      const sanitizedPath = sanitizeInput(filePath, 255);\n      const fullPath = path.isAbsolute(sanitizedPath) \n        ? sanitizedPath \n        : path.join(this.templatesDir, sanitizedPath);\n      \n      const normalizedPath = path.normalize(fullPath);\n      return normalizedPath.startsWith(this.templatesDir);\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Get the element type\n   */\n  getElementType(): ElementType {\n    return ElementType.TEMPLATE;\n  }\n\n  /**\n   * Get the file extension\n   */\n  getFileExtension(): string {\n    return '.md';\n  }\n\n  /**\n   * Find templates by category\n   */\n  async findByCategory(category: string): Promise<Template[]> {\n    const sanitizedCategory = sanitizeInput(category, 50);\n    return this.list().then(templates => \n      templates.filter(t => t.metadata.category === sanitizedCategory)\n    );\n  }\n\n  /**\n   * Find templates by tag\n   */\n  async findByTag(tag: string): Promise<Template[]> {\n    const sanitizedTag = sanitizeInput(tag, 50);\n    return this.list().then(templates => \n      templates.filter(t => t.metadata.tags?.includes(sanitizedTag))\n    );\n  }\n\n  /**\n   * Get most used templates\n   */\n  async getMostUsed(limit: number = 10): Promise<Template[]> {\n    // SECURITY FIX: Validate limit parameter to prevent excessive memory usage\n    // Previously: No validation could allow very large limits\n    // Now: Enforces reasonable bounds\n    const MIN_LIMIT = 1;\n    const MAX_LIMIT = 100;\n    const validatedLimit = Math.max(MIN_LIMIT, Math.min(MAX_LIMIT, Math.floor(limit)));\n    \n    if (limit !== validatedLimit) {\n      logger.warn(`getMostUsed: limit ${limit} adjusted to ${validatedLimit} (valid range: ${MIN_LIMIT}-${MAX_LIMIT})`);\n    }\n    \n    const templates = await this.list();\n    return templates\n      .sort((a, b) => (b.metadata.usage_count || 0) - (a.metadata.usage_count || 0))\n      .slice(0, validatedLimit);\n  }\n}"]}
528
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"TemplateManager.js","sourceRoot":"","sources":["../../../src/elements/templates/TemplateManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * TemplateManager - Implementation of IElementManager for Template elements\n * Handles CRUD operations and lifecycle management for templates implementing IElement\n * \n * SECURITY FIXES IMPLEMENTED (Following PR #319 patterns):\n * 1. CRITICAL: Fixed race conditions in file operations by using FileLockManager for atomic reads/writes\n * 2. CRITICAL: Fixed dynamic require() statements by using static imports\n * 3. HIGH: Fixed unvalidated YAML parsing vulnerability by using SecureYamlParser\n * 4. MEDIUM: All user inputs are now validated and sanitized\n * 5. MEDIUM: Audit logging added for security operations\n * 6. MEDIUM: Path traversal prevention for all file operations\n */\n\nimport { IElementManager } from '../../types/elements/IElementManager.js';\nimport { ElementValidationResult } from '../../types/elements/IElement.js';\nimport { Template, TemplateMetadata } from './Template.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { PortfolioManager } from '../../portfolio/PortfolioManager.js';\nimport { logger } from '../../utils/logger.js';\nimport { FileLockManager } from '../../security/fileLockManager.js';\nimport { SecureYamlParser } from '../../security/secureYamlParser.js';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { sanitizeInput, validatePath } from '../../security/InputValidator.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { promises as fs } from 'fs';\nimport * as path from 'path';\nimport * as yaml from 'js-yaml';\nimport matter from 'gray-matter';\n\nexport class TemplateManager implements IElementManager<Template> {\n  private portfolioManager: PortfolioManager;\n  private templatesDir: string;\n  private templates: Map<string, Template> = new Map();\n\n  constructor() {\n    this.portfolioManager = PortfolioManager.getInstance();\n    this.templatesDir = this.portfolioManager.getElementDir(ElementType.TEMPLATE);\n  }\n\n  /**\n   * Load a template from file\n   * SECURITY FIX #1: Uses FileLockManager.atomicReadFile() instead of fs.readFile()\n   * to prevent race conditions and ensure atomic file operations\n   */\n  async load(filePath: string): Promise<Template> {\n    // SECURITY FIX #4 & #6: Validate and sanitize the file path\n    // Previously: Direct use of user-provided paths could lead to path traversal\n    // Now: Full validation prevents accessing files outside templates directory\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // Ensure the path is within the templates directory\n    const fullPath = path.isAbsolute(sanitizedPath) \n      ? sanitizedPath \n      : path.join(this.templatesDir, sanitizedPath);\n    \n    // SECURITY FIX #6: Prevent path traversal attacks\n    const normalizedPath = path.normalize(fullPath);\n    if (!normalizedPath.startsWith(this.templatesDir)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'CRITICAL',\n        source: 'TemplateManager.load',\n        details: `Attempted to access file outside templates directory: ${sanitizedPath}`\n      });\n      throw new Error('Path traversal attempt detected');\n    }\n\n    try {\n      // CRITICAL FIX: Use atomic file read to prevent race conditions\n      // Previously: const content = await fs.readFile(fullPath, 'utf-8');\n      // Now: Uses FileLockManager with proper encoding object format\n      const content = await FileLockManager.atomicReadFile(normalizedPath, { encoding: 'utf-8' });\n      \n      // Parse the template file (expected format: YAML frontmatter + content)\n      const parsed = matter(content);\n      \n      // SECURITY FIX #3: Validate YAML metadata using SecureYamlParser\n      // Previously: Frontmatter parsing could be vulnerable to YAML injection\n      // Now: SecureYamlParser validates and sanitizes YAML content\n      const metadata = await this.validateMetadata(parsed.data);\n      \n      // Create the template instance\n      const template = new Template(metadata, parsed.content);\n      \n      // Cache the template\n      this.templates.set(normalizedPath, template);\n      \n      // SECURITY FIX #5: Log successful template load for audit trail\n      SecurityMonitor.logSecurityEvent({\n        type: 'TEMPLATE_LOADED',\n        severity: 'LOW',\n        source: 'TemplateManager.load',\n        details: `Template loaded: ${template.metadata.name} from ${path.basename(normalizedPath)}`\n      });\n      \n      return template;\n    } catch (error) {\n      logger.error(`Failed to load template from ${normalizedPath}: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * Save a template to file\n   * SECURITY FIX #1: Uses FileLockManager.atomicWriteFile() for atomic operations\n   */\n  async save(template: Template, filePath: string): Promise<void> {\n    // SECURITY FIX #4: Validate inputs\n    const validation = template.validate();\n    if (!validation.valid) {\n      throw new Error(`Cannot save invalid template: ${validation.errors?.map(e => e.message).join(', ')}`);\n    }\n\n    // SECURITY FIX #4 & #6: Validate and sanitize the file path\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // Ensure the path is within the templates directory\n    const fullPath = path.isAbsolute(sanitizedPath) \n      ? sanitizedPath \n      : path.join(this.templatesDir, sanitizedPath);\n    \n    // SECURITY FIX #6: Prevent path traversal attacks\n    const normalizedPath = path.normalize(fullPath);\n    if (!normalizedPath.startsWith(this.templatesDir)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'CRITICAL',\n        source: 'TemplateManager.save',\n        details: `Attempted to save file outside templates directory: ${sanitizedPath}`\n      });\n      throw new Error('Path traversal attempt detected');\n    }\n\n    try {\n      // Ensure directory exists\n      await fs.mkdir(path.dirname(normalizedPath), { recursive: true });\n      \n      // Create frontmatter content\n      const frontmatter = this.createFrontmatter(template.metadata);\n      const content = `---\\n${frontmatter}\\n---\\n\\n${template.content}`;\n      \n      // CRITICAL FIX: Use atomic file write to prevent race conditions\n      // Previously: await fs.writeFile(fullPath, content, 'utf-8');\n      // Now: Uses FileLockManager for atomic operations\n      await FileLockManager.atomicWriteFile(normalizedPath, content, { encoding: 'utf-8' });\n      \n      // Update cache\n      this.templates.set(normalizedPath, template);\n      \n      // SECURITY FIX #5: Log successful save for audit trail\n      SecurityMonitor.logSecurityEvent({\n        type: 'TEMPLATE_SAVED',\n        severity: 'LOW',\n        source: 'TemplateManager.save',\n        details: `Template saved: ${template.metadata.name} to ${path.basename(normalizedPath)}`\n      });\n      \n      logger.info(`Template saved: ${template.metadata.name}`);\n    } catch (error) {\n      logger.error(`Failed to save template to ${normalizedPath}: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * List all templates\n   * SECURITY FIX: Uses validated directory path\n   */\n  async list(): Promise<Template[]> {\n    try {\n      const files = await fs.readdir(this.templatesDir);\n      const templateFiles = files.filter(file => file.endsWith('.md') || file.endsWith('.yaml'));\n      \n      // Load templates in parallel with error handling\n      const templates = await Promise.all(\n        templateFiles.map(async file => {\n          try {\n            return await this.load(file);\n          } catch (error) {\n            logger.error(`Failed to load template ${file}: ${error}`);\n            return null;\n          }\n        })\n      );\n      \n      // Filter out failed loads\n      return templates.filter((t): t is Template => t !== null);\n    } catch (error) {\n      // Handle missing directory gracefully with type-safe check\n      if (error && typeof error === 'object' && 'code' in error && error.code === 'ENOENT') {\n        logger.debug('Templates directory does not exist yet, returning empty array');\n        return [];\n      }\n      logger.error(`Failed to list templates: ${error}`);\n      return [];\n    }\n  }\n\n  /**\n   * Find a template by predicate\n   */\n  async find(predicate: (template: Template) => boolean): Promise<Template | undefined> {\n    const templates = await this.list();\n    return templates.find(predicate);\n  }\n\n  /**\n   * Create a new template\n   */\n  async create(data: {name: string; description: string; content?: string; metadata?: any}): Promise<Template> {\n    // SECURITY FIX #4: Validate and sanitize all inputs\n    const sanitizedName = sanitizeInput(data.name || 'new-template', 100);\n    const sanitizedDescription = sanitizeInput(data.description || '', 500);\n    const sanitizedContent = sanitizeInput(data.content || '', 100000); // 100KB max\n    \n    // Create the template instance\n    const template = new Template({\n      ...data.metadata,\n      name: sanitizedName,\n      description: sanitizedDescription\n    }, sanitizedContent);\n    \n    // Generate filename from template name\n    const filename = `${sanitizedName.toLowerCase().replace(/[^a-z0-9-]/g, '-')}.md`;\n    \n    // Save the template\n    await this.save(template, filename);\n    \n    // SECURITY FIX #5: Audit successful creation\n    SecurityMonitor.logSecurityEvent({\n      type: 'ELEMENT_CREATED',\n      severity: 'LOW',\n      source: 'TemplateManager.create',\n      details: `Template created: ${template.metadata.name}`\n    });\n    \n    return template;\n  }\n\n  /**\n   * Delete a template\n   * SECURITY FIX #6: Path validation to prevent deletion outside directory\n   */\n  async delete(filePath: string): Promise<void> {\n    // SECURITY FIX #4 & #6: Validate and sanitize the file path\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    \n    // Ensure the path is within the templates directory\n    const fullPath = path.isAbsolute(sanitizedPath) \n      ? sanitizedPath \n      : path.join(this.templatesDir, sanitizedPath);\n    \n    // SECURITY FIX #6: Prevent path traversal attacks\n    const normalizedPath = path.normalize(fullPath);\n    if (!normalizedPath.startsWith(this.templatesDir)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'CRITICAL',\n        source: 'TemplateManager.delete',\n        details: `Attempted to delete file outside templates directory: ${sanitizedPath}`\n      });\n      throw new Error('Path traversal attempt detected');\n    }\n\n    try {\n      await fs.unlink(normalizedPath);\n      this.templates.delete(normalizedPath);\n      \n      // SECURITY FIX #5: Log deletion for audit trail\n      SecurityMonitor.logSecurityEvent({\n        type: 'TEMPLATE_DELETED',\n        severity: 'MEDIUM',\n        source: 'TemplateManager.delete',\n        details: `Template deleted: ${path.basename(normalizedPath)}`\n      });\n      \n      logger.info(`Template deleted: ${path.basename(normalizedPath)}`);\n    } catch (error) {\n      logger.error(`Failed to delete template ${normalizedPath}: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * Import a template from external format\n   * SECURITY FIX #3: Uses SecureYamlParser for safe YAML parsing\n   */\n  async importElement(data: string, format: 'json' | 'yaml' | 'markdown'): Promise<Template> {\n    try {\n      let metadata: Partial<TemplateMetadata>;\n      let content: string = '';\n      \n      switch (format) {\n        case 'json':\n          const jsonData = JSON.parse(data);\n          metadata = await this.validateMetadata(jsonData.metadata || {});\n          content = jsonData.content || '';\n          break;\n          \n        case 'yaml':\n          // HIGH SEVERITY FIX: Use SecureYamlParser to prevent YAML injection attacks\n          // Previously: Used unsafe YAML parsing without validation\n          // Now: Uses SecureYamlParser which validates content and prevents malicious patterns\n          const parsed = SecureYamlParser.parse(data, {\n            maxYamlSize: 64 * 1024, // 64KB limit\n            validateContent: true\n          });\n          \n          metadata = await this.validateMetadata((parsed as any).metadata || {});\n          content = parsed.content || '';\n          \n          // Log security event for audit trail\n          SecurityMonitor.logSecurityEvent({\n            type: 'YAML_PARSE_SUCCESS',\n            severity: 'LOW',\n            source: 'TemplateManager.importElement',\n            details: 'YAML content safely parsed during import'\n          });\n          break;\n          \n        case 'markdown':\n          const mdParsed = matter(data);\n          metadata = await this.validateMetadata(mdParsed.data);\n          content = mdParsed.content;\n          break;\n          \n        default:\n          throw new Error(`Unsupported import format: ${format}`);\n      }\n      \n      // Create and validate the template\n      const template = new Template(metadata, content);\n      const validation = template.validate();\n      \n      if (!validation.valid) {\n        throw new Error(`Invalid template: ${validation.errors?.map(e => e.message).join(', ')}`);\n      }\n      \n      return template;\n    } catch (error) {\n      logger.error(`Failed to import template: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * Export a template to external format\n   * SECURITY FIX #3: Uses safe YAML serialization\n   */\n  async exportElement(template: Template, format: 'json' | 'yaml' | 'markdown'): Promise<string> {\n    try {\n      switch (format) {\n        case 'json':\n          return template.serialize();\n          \n        case 'yaml':\n          // SECURITY FIX: Use yaml.dump with FAILSAFE_SCHEMA to prevent code execution\n          // Previously: Could potentially use unsafe YAML features\n          // Now: FAILSAFE_SCHEMA only allows basic YAML types, no JS-specific constructs\n          const yamlData = {\n            metadata: template.metadata,\n            content: template.content,\n            id: template.id,\n            version: template.version\n          };\n          \n          return yaml.dump(yamlData, {\n            // SECURITY TRADE-OFF: Using DEFAULT_SCHEMA instead of FAILSAFE_SCHEMA\n            // Reason: FAILSAFE_SCHEMA doesn't support number types which are needed for template metadata\n            // Risk: DEFAULT_SCHEMA allows more YAML features that could be exploited\n            // Mitigation: noRefs prevents reference attacks, skipInvalid drops dangerous constructs\n            // Consider: For maximum security, implement custom schema that only allows needed types\n            schema: yaml.DEFAULT_SCHEMA,\n            noRefs: true,        // Prevent reference attacks\n            sortKeys: true,      // Consistent output\n            skipInvalid: true,   // Skip unserializable values instead of throwing\n            condenseFlow: true,  // More compact output\n            quotingType: '\"',    // Force double quotes for consistency\n            forceQuotes: false   // Only quote when necessary\n          });\n          \n        case 'markdown':\n          const frontmatter = this.createFrontmatter(template.metadata);\n          return `---\\n${frontmatter}\\n---\\n\\n${template.content}`;\n          \n        default:\n          throw new Error(`Unsupported export format: ${format}`);\n      }\n    } catch (error) {\n      logger.error(`Failed to export template: ${error}`);\n      throw error;\n    }\n  }\n\n  /**\n   * Validate and sanitize metadata\n   * SECURITY FIX #4: Comprehensive metadata validation\n   */\n  private async validateMetadata(data: any): Promise<Partial<TemplateMetadata>> {\n    const metadata: Partial<TemplateMetadata> = {};\n    \n    // Sanitize string fields\n    if (data.name) {\n      metadata.name = sanitizeInput(UnicodeValidator.normalize(data.name).normalizedContent, 100);\n    }\n    \n    if (data.description) {\n      metadata.description = sanitizeInput(UnicodeValidator.normalize(data.description).normalizedContent, 500);\n    }\n    \n    if (data.category) {\n      metadata.category = sanitizeInput(data.category, 50);\n    }\n    \n    if (data.output_format) {\n      metadata.output_format = sanitizeInput(data.output_format, 20);\n    }\n    \n    // Validate arrays\n    if (Array.isArray(data.tags)) {\n      metadata.tags = data.tags.map((tag: any) => sanitizeInput(String(tag), 50));\n    }\n    \n    if (Array.isArray(data.includes)) {\n      metadata.includes = data.includes.map((inc: any) => sanitizeInput(String(inc), 200));\n    }\n    \n    // Copy safe fields\n    if (typeof data.usage_count === 'number') {\n      metadata.usage_count = Math.max(0, Math.floor(data.usage_count));\n    }\n    \n    if (data.last_used) {\n      metadata.last_used = sanitizeInput(String(data.last_used), 50);\n    }\n    \n    // Validate complex fields\n    if (Array.isArray(data.variables)) {\n      metadata.variables = data.variables.map((v: any) => ({\n        name: sanitizeInput(v.name || '', 50),\n        type: sanitizeInput(v.type || 'string', 20),\n        description: v.description ? sanitizeInput(v.description, 200) : undefined,\n        required: Boolean(v.required),\n        default: v.default,\n        validation: v.validation ? sanitizeInput(v.validation, 200) : undefined,\n        options: Array.isArray(v.options) ? v.options.map((o: any) => sanitizeInput(String(o), 100)) : undefined,\n        format: v.format ? sanitizeInput(v.format, 50) : undefined\n      }));\n    }\n    \n    if (Array.isArray(data.examples)) {\n      metadata.examples = data.examples.map((ex: any) => ({\n        title: sanitizeInput(ex.title || '', 100),\n        description: ex.description ? sanitizeInput(ex.description, 500) : undefined,\n        variables: ex.variables || {},\n        output: ex.output ? sanitizeInput(ex.output, 5000) : undefined\n      }));\n    }\n    \n    // Copy standard element fields\n    metadata.author = data.author ? sanitizeInput(data.author, 100) : undefined;\n    metadata.version = data.version ? sanitizeInput(data.version, 20) : undefined;\n    \n    return metadata;\n  }\n\n  /**\n   * Create YAML frontmatter from metadata\n   * SECURITY FIX #3: Safe YAML generation\n   */\n  private createFrontmatter(metadata: TemplateMetadata): string {\n    // SECURITY FIX: Use yaml.dump with security options\n    // Ensures no code execution vulnerabilities in generated YAML\n    const safeMetadata = {\n      name: metadata.name,\n      description: metadata.description,\n      author: metadata.author,\n      version: metadata.version,\n      category: metadata.category,\n      output_format: metadata.output_format,\n      tags: metadata.tags,\n      includes: metadata.includes,\n      usage_count: metadata.usage_count,\n      last_used: metadata.last_used,\n      variables: metadata.variables,\n      examples: metadata.examples\n    };\n    \n    // Remove undefined values\n    const cleanMetadata = Object.fromEntries(\n      Object.entries(safeMetadata).filter(([_, value]) => value !== undefined)\n    );\n    \n    return yaml.dump(cleanMetadata, {\n      // SECURITY TRADE-OFF: Same as exportElement - using DEFAULT_SCHEMA for type support\n      // See exportElement method for detailed security considerations\n      schema: yaml.DEFAULT_SCHEMA,\n      noRefs: true,\n      sortKeys: true,\n      lineWidth: 80,\n      skipInvalid: true,\n      condenseFlow: true,\n      quotingType: '\"',\n      forceQuotes: false\n    });\n  }\n\n  /**\n   * Check if a template file exists\n   */\n  async exists(filePath: string): Promise<boolean> {\n    const sanitizedPath = sanitizeInput(filePath, 255);\n    const fullPath = path.isAbsolute(sanitizedPath) \n      ? sanitizedPath \n      : path.join(this.templatesDir, sanitizedPath);\n    \n    try {\n      await fs.access(fullPath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Find multiple templates by predicate\n   */\n  async findMany(predicate: (template: Template) => boolean): Promise<Template[]> {\n    const templates = await this.list();\n    return templates.filter(predicate);\n  }\n\n  /**\n   * Validate a template\n   */\n  validate(template: Template): ElementValidationResult {\n    return template.validate();\n  }\n\n  /**\n   * Validate a file path\n   */\n  validatePath(filePath: string): boolean {\n    try {\n      const sanitizedPath = sanitizeInput(filePath, 255);\n      const fullPath = path.isAbsolute(sanitizedPath) \n        ? sanitizedPath \n        : path.join(this.templatesDir, sanitizedPath);\n      \n      const normalizedPath = path.normalize(fullPath);\n      return normalizedPath.startsWith(this.templatesDir);\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Get the element type\n   */\n  getElementType(): ElementType {\n    return ElementType.TEMPLATE;\n  }\n\n  /**\n   * Get the file extension\n   */\n  getFileExtension(): string {\n    return '.md';\n  }\n\n  /**\n   * Find templates by category\n   */\n  async findByCategory(category: string): Promise<Template[]> {\n    const sanitizedCategory = sanitizeInput(category, 50);\n    return this.list().then(templates => \n      templates.filter(t => t.metadata.category === sanitizedCategory)\n    );\n  }\n\n  /**\n   * Find templates by tag\n   */\n  async findByTag(tag: string): Promise<Template[]> {\n    const sanitizedTag = sanitizeInput(tag, 50);\n    return this.list().then(templates => \n      templates.filter(t => t.metadata.tags?.includes(sanitizedTag))\n    );\n  }\n\n  /**\n   * Get most used templates\n   */\n  async getMostUsed(limit: number = 10): Promise<Template[]> {\n    // SECURITY FIX: Validate limit parameter to prevent excessive memory usage\n    // Previously: No validation could allow very large limits\n    // Now: Enforces reasonable bounds\n    const MIN_LIMIT = 1;\n    const MAX_LIMIT = 100;\n    const validatedLimit = Math.max(MIN_LIMIT, Math.min(MAX_LIMIT, Math.floor(limit)));\n    \n    if (limit !== validatedLimit) {\n      logger.warn(`getMostUsed: limit ${limit} adjusted to ${validatedLimit} (valid range: ${MIN_LIMIT}-${MAX_LIMIT})`);\n    }\n    \n    const templates = await this.list();\n    return templates\n      .sort((a, b) => (b.metadata.usage_count || 0) - (a.metadata.usage_count || 0))\n      .slice(0, validatedLimit);\n  }\n}"]}
package/dist/index.d.ts CHANGED
@@ -10,7 +10,6 @@ export declare class DollhouseMCPServer implements IToolHandler {
10
10
  private apiCache;
11
11
  private rateLimitTracker;
12
12
  private indicatorConfig;
13
- private personaManager;
14
13
  private githubClient;
15
14
  private collectionBrowser;
16
15
  private collectionSearch;
@@ -30,6 +29,16 @@ export declare class DollhouseMCPServer implements IToolHandler {
30
29
  constructor();
31
30
  private initializePortfolio;
32
31
  private getPersonaIndicator;
32
+ /**
33
+ * Normalize element type to handle both singular (new) and plural (legacy) forms
34
+ * This provides backward compatibility during the transition to v1.4.0
35
+ */
36
+ private normalizeElementType;
37
+ /**
38
+ * Sanitize metadata object to prevent prototype pollution
39
+ * Removes any dangerous properties that could affect Object.prototype
40
+ */
41
+ private sanitizeMetadata;
33
42
  private loadPersonas;
34
43
  listPersonas(): Promise<{
35
44
  content: {
@@ -115,6 +124,49 @@ export declare class DollhouseMCPServer implements IToolHandler {
115
124
  text: string;
116
125
  }[];
117
126
  }>;
127
+ createElement(args: {
128
+ name: string;
129
+ type: string;
130
+ description: string;
131
+ content?: string;
132
+ metadata?: Record<string, any>;
133
+ }): Promise<{
134
+ content: {
135
+ type: string;
136
+ text: string;
137
+ }[];
138
+ }>;
139
+ editElement(args: {
140
+ name: string;
141
+ type: string;
142
+ field: string;
143
+ value: string | number | boolean | Record<string, any> | any[];
144
+ }): Promise<{
145
+ content: {
146
+ type: string;
147
+ text: string;
148
+ }[];
149
+ }>;
150
+ validateElement(args: {
151
+ name: string;
152
+ type: string;
153
+ strict?: boolean;
154
+ }): Promise<{
155
+ content: {
156
+ type: string;
157
+ text: string;
158
+ }[];
159
+ }>;
160
+ deleteElement(args: {
161
+ name: string;
162
+ type: string;
163
+ deleteData?: boolean;
164
+ }): Promise<{
165
+ content: {
166
+ type: string;
167
+ text: string;
168
+ }[];
169
+ }>;
118
170
  browseCollection(section?: string, type?: string): Promise<{
119
171
  content: {
120
172
  type: string;
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AASA,OAAO,EAA8D,KAAK,eAAe,EAAE,MAAM,8BAA8B,CAAC;AAkBhI,OAAO,EAAe,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAY9D,qBAAa,kBAAmB,YAAW,YAAY;IACrD,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,QAAQ,CAAmC;IACnD,OAAO,CAAC,aAAa,CAAuB;IAC5C,OAAO,CAAC,WAAW,CAAuB;IAC1C,OAAO,CAAC,QAAQ,CAA4B;IAC5C,OAAO,CAAC,gBAAgB,CAA+B;IACvD,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,cAAc,CAAiB;IACvC,OAAO,CAAC,YAAY,CAAe;IACnC,OAAO,CAAC,iBAAiB,CAAoB;IAC7C,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,cAAc,CAAiB;IACvC,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,aAAa,CAAgB;IACrC,OAAO,CAAC,WAAW,CAAc;IACjC,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,aAAa,CAAgB;IACrC,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,YAAY,CAAe;IACnC,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,YAAY,CAAe;;YAwErB,mBAAmB;IA8BjC,OAAO,CAAC,mBAAmB;YAkBb,YAAY;IAqFpB,YAAY;;;;;;IAiCZ,eAAe,CAAC,iBAAiB,EAAE,MAAM;;;;;;IAmCzC,gBAAgB;;;;;;IAuChB,iBAAiB;;;;;;IAiBjB,iBAAiB,CAAC,iBAAiB,EAAE,MAAM;;;;;;IAkC3C,cAAc;;;;;;IAcd,YAAY,CAAC,IAAI,EAAE,MAAM;;;;;;IAmGzB,eAAe,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;;;;;;IAyF1C,iBAAiB,CAAC,IAAI,EAAE,MAAM;;;;;;IAkF9B,iBAAiB,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;;;;;;IA0E5C,iBAAiB,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;;;;;;IA6I5C,cAAc,CAAC,IAAI,EAAE,MAAM;;;;;;IA2D3B,cAAc,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;;;;;;IAmC3D,YAAY,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;;;;;;IAuCvC,gBAAgB,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM;;;;;;IAyChD,gBAAgB,CAAC,KAAK,EAAE,MAAM;;;;;;IA6B9B,oBAAoB,CAAC,IAAI,EAAE,MAAM;;;;;;IA0BjC,cAAc,CAAC,SAAS,EAAE,MAAM;;;;;;IA+ChC,aAAa,CAAC,iBAAiB,EAAE,MAAM;;;;;;IAqFvC,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM;;;;;;IA8DhD,eAAe;;;;;;IAuCf,iBAAiB;;;;;;IAwBvB,OAAO,CAAC,4BAA4B;IAK9B,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM;;;;;;IAmM1G,WAAW,CAAC,iBAAiB,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM;;;;;;IAyNnE,eAAe,CAAC,iBAAiB,EAAE,MAAM;;;;;;IAyIzC,eAAe;;;;;;IASf,YAAY,CAAC,OAAO,EAAE,OAAO;;;;;;IA+B7B,cAAc,CAAC,OAAO,EAAE,OAAO;;;;;;IAW/B,eAAe;;;;;;IAqBrB;;OAEG;IACG,kBAAkB,CAAC,MAAM,EAAE,OAAO,CAAC,eAAe,CAAC;;;;;;IAwGzD;;OAEG;IACG,kBAAkB;;;;;;IA4DxB;;OAEG;IACG,aAAa,CAAC,WAAW,EAAE,MAAM;;;;;;IAqCvC;;OAEG;IACG,iBAAiB,CAAC,eAAe,UAAO;;;;;;IAuB9C;;OAEG;IACG,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,UAAQ;;;;;;IAgCrD;;OAEG;IACG,YAAY,CAAC,WAAW,EAAE,MAAM,EAAE,UAAU,SAAI;;;;;;IAuCtD;;OAEG;IACG,aAAa,CAAC,GAAG,EAAE,MAAM,EAAE,SAAS,UAAQ;;;;;;IAmD5C,GAAG;CAQV"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAOA,OAAO,EAA8D,KAAK,eAAe,EAAE,MAAM,8BAA8B,CAAC;AAgBhI,OAAO,EAAe,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAY9D,qBAAa,kBAAmB,YAAW,YAAY;IACrD,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,QAAQ,CAAmC;IACnD,OAAO,CAAC,aAAa,CAAuB;IAC5C,OAAO,CAAC,WAAW,CAAuB;IAC1C,OAAO,CAAC,QAAQ,CAA4B;IAC5C,OAAO,CAAC,gBAAgB,CAA+B;IACvD,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,YAAY,CAAe;IACnC,OAAO,CAAC,iBAAiB,CAAoB;IAC7C,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,cAAc,CAAiB;IACvC,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,aAAa,CAAgB;IACrC,OAAO,CAAC,WAAW,CAAc;IACjC,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,aAAa,CAAgB;IACrC,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,YAAY,CAAe;IACnC,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,YAAY,CAAe;;YAuErB,mBAAmB;IA8BjC,OAAO,CAAC,mBAAmB;IAkB3B;;;OAGG;IACH,OAAO,CAAC,oBAAoB;IA2B5B;;;OAGG;IACH,OAAO,CAAC,gBAAgB;YAsBV,YAAY;IAqFpB,YAAY;;;;;;IAiCZ,eAAe,CAAC,iBAAiB,EAAE,MAAM;;;;;;IAmCzC,gBAAgB;;;;;;IAuChB,iBAAiB;;;;;;IAiBjB,iBAAiB,CAAC,iBAAiB,EAAE,MAAM;;;;;;IAkC3C,cAAc;;;;;;IAcd,YAAY,CAAC,IAAI,EAAE,MAAM;;;;;;IAsGzB,eAAe,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;;;;;;IA4F1C,iBAAiB,CAAC,IAAI,EAAE,MAAM;;;;;;IAqF9B,iBAAiB,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;;;;;;IA6E5C,iBAAiB,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;;;;;;IAgJ5C,cAAc,CAAC,IAAI,EAAE,MAAM;;;;;;IA8D3B,cAAc,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC;;;;;;IAmC3D,YAAY,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM;;;;;;IAoCvC,aAAa,CAAC,IAAI,EAAE;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAC;QAAC,OAAO,CAAC,EAAE,MAAM,CAAC;QAAC,QAAQ,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;KAAC;;;;;;IAsGvH,WAAW,CAAC,IAAI,EAAE;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,GAAG,EAAE,CAAA;KAAC;;;;;;IAiJ7H,eAAe,CAAC,IAAI,EAAE;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,CAAC,EAAE,OAAO,CAAA;KAAC;;;;;;IA6GpE,aAAa,CAAC,IAAI,EAAE;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,MAAM,CAAC;QAAC,UAAU,CAAC,EAAE,OAAO,CAAA;KAAC;;;;;;IA2NtE,gBAAgB,CAAC,OAAO,CAAC,EAAE,MAAM,EAAE,IAAI,CAAC,EAAE,MAAM;;;;;;IAyChD,gBAAgB,CAAC,KAAK,EAAE,MAAM;;;;;;IA6B9B,oBAAoB,CAAC,IAAI,EAAE,MAAM;;;;;;IA0BjC,cAAc,CAAC,SAAS,EAAE,MAAM;;;;;;IA+ChC,aAAa,CAAC,iBAAiB,EAAE,MAAM;;;;;;IAqFvC,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,MAAM;;;;;;IA8DhD,eAAe;;;;;;IAuCf,iBAAiB;;;;;;IAwBvB,OAAO,CAAC,4BAA4B;IAK9B,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM;;;;;;IAmM1G,WAAW,CAAC,iBAAiB,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM;;;;;;IAyNnE,eAAe,CAAC,iBAAiB,EAAE,MAAM;;;;;;IAyIzC,eAAe;;;;;;IASf,YAAY,CAAC,OAAO,EAAE,OAAO;;;;;;IA+B7B,cAAc,CAAC,OAAO,EAAE,OAAO;;;;;;IAW/B,eAAe;;;;;;IAqBrB;;OAEG;IACG,kBAAkB,CAAC,MAAM,EAAE,OAAO,CAAC,eAAe,CAAC;;;;;;IAwGzD;;OAEG;IACG,kBAAkB;;;;;;IA4DxB;;OAEG;IACG,aAAa,CAAC,WAAW,EAAE,MAAM;;;;;;IAqCvC;;OAEG;IACG,iBAAiB,CAAC,eAAe,UAAO;;;;;;IAuB9C;;OAEG;IACG,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,UAAQ;;;;;;IAgCrD;;OAEG;IACG,YAAY,CAAC,WAAW,EAAE,MAAM,EAAE,UAAU,SAAI;;;;;;IAuCtD;;OAEG;IACG,aAAa,CAAC,GAAG,EAAE,MAAM,EAAE,SAAS,UAAQ;;;;;;IAmD5C,GAAG;CAQV"}