@dollhousemcp/mcp-server 1.9.6 → 1.9.7

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
+ ## [1.9.7] - 2025-09-20
4
+
5
+ ### Fixed
6
+ - **NPM Package Build**: Corrected v1.9.6 NPM package which was built from wrong commit
7
+ - The v1.9.6 tag was created before the memory display fixes were merged
8
+ - This resulted in the NPM package missing the critical memory content display fix
9
+ - v1.9.7 includes all fixes that were intended for v1.9.6
10
+ - Memory elements now correctly display their content instead of "No content stored"
11
+
12
+ ### Note
13
+ This release republishes v1.9.6 with the correct code. The memory display fix (PR #1036) and other improvements were merged to main before the v1.9.6 release but the NPM package was accidentally built from an earlier commit.
14
+
3
15
  ## [1.9.6] - 2025-09-20
4
16
 
5
17
  ### 🎉 First External Contribution
package/README.github.md CHANGED
@@ -873,6 +873,74 @@ For detailed guidelines, see [CONTRIBUTING.md](CONTRIBUTING.md).
873
873
 
874
874
  ## 🏷️ Version History
875
875
 
876
+ ### v1.9.6 - September 20, 2025
877
+
878
+ **🎉 First External Contribution**: Performance and security improvements from the community!
879
+
880
+ #### ✨ Highlights
881
+ - **Fixed**: Memory display bug - added content getter to Memory class (PR #1036)
882
+ - **Fixed**: Flaky macOS tests on Node 22+ (PR #1038)
883
+ - **Enhanced**: Optimized whitespace detection for better performance (PR #1037)
884
+ - **Security**: Strengthened path traversal protection (PR #1037)
885
+ - **Attribution**: Thanks to @jeetsingh008 for identifying improvements!
886
+
887
+ ---
888
+
889
+ ### v1.9.5 - September 19, 2025
890
+
891
+ **Memory YAML Parsing Fix**: Resolved display issues with pure YAML memory files
892
+
893
+ #### 🔧 Bug Fixes
894
+ - **Fixed**: Memory files showing incorrect names for pure YAML format
895
+ - **Enhanced**: Added comprehensive test coverage for memory file formats
896
+ - **Technical**: Improved compatibility between SecureYamlParser and pure YAML
897
+
898
+ ---
899
+
900
+ ### v1.9.4 - September 19, 2025
901
+
902
+ **Memory Name Display Fix**: Corrected "Unnamed Memory" display issue
903
+
904
+ #### 🔧 Bug Fixes
905
+ - **Fixed**: Memory elements showing as "Unnamed Memory" in list output
906
+ - **Fixed**: Corrected metadata parsing path in SecureYamlParser
907
+ - **Technical**: Added retention format parsing for various formats
908
+
909
+ ---
910
+
911
+ ### v1.9.3 - September 19, 2025
912
+
913
+ **Memory Element MCP Support**: Complete MCP tool handler integration
914
+
915
+ #### 🔧 Bug Fixes
916
+ - **Fixed**: Added Memory element support to all MCP tool handlers
917
+ - **Fixed**: Resolved "Unknown element type 'memories'" errors
918
+ - **Technical**: Added Memory case handling to 8 critical methods
919
+
920
+ ---
921
+
922
+ ### v1.9.2 - September 19, 2025
923
+
924
+ **Branch Synchronization**: Documentation and configuration alignment
925
+
926
+ #### 🔧 Improvements
927
+ - **Fixed**: Resolved divergence between main and develop branches
928
+ - **Enhanced**: Updated documentation to reflect all features
929
+ - **Technical**: Merged 58 commits from develop branch
930
+
931
+ ---
932
+
933
+ ### v1.9.1 - September 19, 2025
934
+
935
+ **Memory Element Hotfix**: Fixed validation and tool descriptions
936
+
937
+ #### 🔧 Bug Fixes
938
+ - **Fixed**: Added 'memories' to validation arrays
939
+ - **Fixed**: Updated collection tool descriptions
940
+ - **Technical**: Clean hotfix for memory element support
941
+
942
+ ---
943
+
876
944
  ### v1.9.0 - September 19, 2025
877
945
 
878
946
  **🎉 Memory Element Release**: Persistent context storage with enterprise-grade features
package/README.md.backup CHANGED
@@ -873,6 +873,74 @@ For detailed guidelines, see [CONTRIBUTING.md](CONTRIBUTING.md).
873
873
 
874
874
  ## 🏷️ Version History
875
875
 
876
+ ### v1.9.6 - September 20, 2025
877
+
878
+ **🎉 First External Contribution**: Performance and security improvements from the community!
879
+
880
+ #### ✨ Highlights
881
+ - **Fixed**: Memory display bug - added content getter to Memory class (PR #1036)
882
+ - **Fixed**: Flaky macOS tests on Node 22+ (PR #1038)
883
+ - **Enhanced**: Optimized whitespace detection for better performance (PR #1037)
884
+ - **Security**: Strengthened path traversal protection (PR #1037)
885
+ - **Attribution**: Thanks to @jeetsingh008 for identifying improvements!
886
+
887
+ ---
888
+
889
+ ### v1.9.5 - September 19, 2025
890
+
891
+ **Memory YAML Parsing Fix**: Resolved display issues with pure YAML memory files
892
+
893
+ #### 🔧 Bug Fixes
894
+ - **Fixed**: Memory files showing incorrect names for pure YAML format
895
+ - **Enhanced**: Added comprehensive test coverage for memory file formats
896
+ - **Technical**: Improved compatibility between SecureYamlParser and pure YAML
897
+
898
+ ---
899
+
900
+ ### v1.9.4 - September 19, 2025
901
+
902
+ **Memory Name Display Fix**: Corrected "Unnamed Memory" display issue
903
+
904
+ #### 🔧 Bug Fixes
905
+ - **Fixed**: Memory elements showing as "Unnamed Memory" in list output
906
+ - **Fixed**: Corrected metadata parsing path in SecureYamlParser
907
+ - **Technical**: Added retention format parsing for various formats
908
+
909
+ ---
910
+
911
+ ### v1.9.3 - September 19, 2025
912
+
913
+ **Memory Element MCP Support**: Complete MCP tool handler integration
914
+
915
+ #### 🔧 Bug Fixes
916
+ - **Fixed**: Added Memory element support to all MCP tool handlers
917
+ - **Fixed**: Resolved "Unknown element type 'memories'" errors
918
+ - **Technical**: Added Memory case handling to 8 critical methods
919
+
920
+ ---
921
+
922
+ ### v1.9.2 - September 19, 2025
923
+
924
+ **Branch Synchronization**: Documentation and configuration alignment
925
+
926
+ #### 🔧 Improvements
927
+ - **Fixed**: Resolved divergence between main and develop branches
928
+ - **Enhanced**: Updated documentation to reflect all features
929
+ - **Technical**: Merged 58 commits from develop branch
930
+
931
+ ---
932
+
933
+ ### v1.9.1 - September 19, 2025
934
+
935
+ **Memory Element Hotfix**: Fixed validation and tool descriptions
936
+
937
+ #### 🔧 Bug Fixes
938
+ - **Fixed**: Added 'memories' to validation arrays
939
+ - **Fixed**: Updated collection tool descriptions
940
+ - **Technical**: Clean hotfix for memory element support
941
+
942
+ ---
943
+
876
944
  ### v1.9.0 - September 19, 2025
877
945
 
878
946
  **🎉 Memory Element Release**: Persistent context storage with enterprise-grade features
@@ -134,6 +134,11 @@ export declare class Memory extends BaseElement implements IElement {
134
134
  * SECURITY: Validates permissions and logs deletion
135
135
  */
136
136
  deleteEntry(id: string): Promise<boolean>;
137
+ /**
138
+ * Get formatted content of all memory entries
139
+ * Returns entries as a readable string for display
140
+ */
141
+ get content(): string;
137
142
  /**
138
143
  * Enforce retention policy by removing expired entries
139
144
  * SECURITY: Ensures memory doesn't grow unbounded
@@ -1 +1 @@
1
- {"version":3,"file":"Memory.d.ts","sourceRoot":"","sources":["../../../src/elements/memories/Memory.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"Memory.d.ts","sourceRoot":"","sources":["../../../src/elements/memories/Memory.ts"],"names":[],"mappings":"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"}
@@ -332,6 +332,23 @@ export class Memory extends BaseElement {
332
332
  this._isDirty = true;
333
333
  return true;
334
334
  }
335
+ /**
336
+ * Get formatted content of all memory entries
337
+ * Returns entries as a readable string for display
338
+ */
339
+ get content() {
340
+ if (this.entries.size === 0) {
341
+ return 'No content stored';
342
+ }
343
+ // Format entries as readable content (newest first)
344
+ const sortedEntries = Array.from(this.entries.values())
345
+ .sort((a, b) => b.timestamp.getTime() - a.timestamp.getTime());
346
+ return sortedEntries.map(entry => {
347
+ const timestamp = entry.timestamp.toISOString();
348
+ const tags = entry.tags && entry.tags.length > 0 ? ` [${entry.tags.join(', ')}]` : '';
349
+ return `[${timestamp}]${tags}: ${entry.content}`;
350
+ }).join('\n\n');
351
+ }
335
352
  /**
336
353
  * Enforce retention policy by removing expired entries
337
354
  * SECURITY: Ensures memory doesn't grow unbounded
@@ -624,4 +641,4 @@ export class Memory extends BaseElement {
624
641
  throw lastError || new Error('Failed to build search index');
625
642
  }
626
643
  }
627
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Memory.js","sourceRoot":"","sources":["../../../src/elements/memories/Memory.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Memory Element - Persistent context storage for continuity and learning\n * \n * Provides multiple storage backends, retention policies, and search capabilities\n * for maintaining context across sessions and interactions.\n * \n * SECURITY MEASURES IMPLEMENTED:\n * 1. Input sanitization for all memory content\n * 2. Memory size limits to prevent unbounded growth\n * 3. Path validation for file-based storage\n * 4. Retention policy enforcement\n * 5. Privacy level access control\n * 6. Audit logging for all operations\n */\n\nimport { BaseElement } from '../BaseElement.js';\nimport { IElement, ElementValidationResult, ValidationError } from '../../types/elements/index.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { IElementMetadata } from '../../types/elements/IElement.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport crypto from 'crypto';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { sanitizeInput } from '../../security/InputValidator.js';\nimport { MEMORY_CONSTANTS, MEMORY_SECURITY_EVENTS, PrivacyLevel, StorageBackend } from './constants.js';\nimport { generateMemoryId } from './utils.js';\nimport { MemorySearchIndex, SearchQuery, SearchIndexConfig } from './MemorySearchIndex.js';\nimport { logger } from '../../utils/logger.js';\nimport DOMPurify from 'dompurify';\nimport { JSDOM } from 'jsdom';\n\n// Initialize DOMPurify with JSDOM\nconst window = new JSDOM('').window;\nconst purify = DOMPurify(window as any);\n\n// Configure DOMPurify for memory content - strip all HTML but keep text\npurify.setConfig({\n  ALLOWED_TAGS: [],  // No HTML tags allowed\n  ALLOWED_ATTR: [],  // No attributes allowed\n  KEEP_CONTENT: true // Keep text content\n});\n\n/**\n * Sanitize content for memory storage\n * More permissive than sanitizeInput - allows punctuation, quotes, etc.\n * but still prevents XSS and control characters\n */\nfunction sanitizeMemoryContent(content: string, maxLength: number): string {\n  if (!content || typeof content !== 'string') {\n    return '';\n  }\n  \n  // First normalize Unicode\n  const normalized = UnicodeValidator.normalize(content).normalizedContent;\n  \n  // Use DOMPurify to strip any HTML/XSS attempts but keep text\n  const cleaned = purify.sanitize(normalized);\n  \n  // Remove only control characters and null bytes\n  return cleaned\n    .replace(/[\\x00-\\x08\\x0B\\x0C\\x0E-\\x1F\\x7F]/g, '') // Remove control chars except \\t \\n \\r\n    .substring(0, maxLength)\n    .trim();\n}\n\nexport interface MemoryMetadata extends IElementMetadata {\n  storageBackend?: StorageBackend;\n  retentionDays?: number;\n  privacyLevel?: PrivacyLevel;\n  searchable?: boolean;\n  maxEntries?: number;\n  encryptionEnabled?: boolean;\n  // Search index configuration (Issue #984)\n  indexThreshold?: number;\n  enableContentIndex?: boolean;\n  maxTermsPerEntry?: number;\n  minTermLength?: number;\n}\n\nexport interface MemoryEntry {\n  id: string;\n  timestamp: Date;\n  content: string;\n  tags?: string[];\n  metadata?: Record<string, any>;\n  expiresAt?: Date;\n  privacyLevel?: PrivacyLevel;\n}\n\nexport interface MemorySearchOptions {\n  query?: string;\n  tags?: string[];\n  startDate?: Date;\n  endDate?: Date;\n  limit?: number;\n  privacyLevel?: PrivacyLevel;\n}\n\n/**\n * Memory Element Implementation\n *\n * TODO: Memory Sharding Strategy (Issue #981)\n * ---------------------------------------------\n * Current: Single Map<id, entry> for all memories\n * Problem: Large memory sets (>10K entries) cause performance degradation\n *\n * Planned Sharding Architecture:\n * 1. Shard memories across multiple files based on hash(memoryId) % shardCount\n * 2. Each shard file <256KB for optimal YAML parsing\n * 3. Memory references stored separately from content (like git objects)\n * 4. Large binary content (PDFs, images) stored as external references\n *\n * Benefits:\n * - Parallel loading of memory shards\n * - Reduced memory footprint (load only needed shards)\n * - Better corruption resistance (one shard failure doesn't affect others)\n * - Efficient incremental updates\n *\n * TODO: Content Integrity Verification (Issue #982)\n * --------------------------------------------------\n * Add SHA-256 hashes to detect:\n * - Accidental corruption from disk errors\n * - Intentional tampering with memory files\n * - Version conflicts during concurrent access\n *\n * Implementation:\n * - Store hash in memory metadata\n * - Verify on load, warn on mismatch\n * - Option to auto-restore from backup on corruption\n *\n * TODO: Memory Capacity Management (Issue #983)\n * ---------------------------------------------\n * Current: Synchronous retention enforcement on each add\n * Better: Background cleanup with smart triggers:\n * - Cleanup when 90% capacity reached\n * - Batch deletions for efficiency\n * - LRU eviction with access tracking\n * - Preserve \"pinned\" memories regardless of age\n */\nexport class Memory extends BaseElement implements IElement {\n  // Memory-specific properties\n  private entries: Map<string, MemoryEntry> = new Map();\n  private storageBackend: StorageBackend;\n  private retentionDays: number;\n  private privacyLevel: PrivacyLevel;\n  private searchable: boolean;\n  private maxEntries: number;\n\n  // Search index for performance (Issue #984)\n  private searchIndex: MemorySearchIndex;\n\n  // Sanitization cache to avoid redundant processing\n  private sanitizationCache: Map<string, string> = new Map();\n  \n  constructor(metadata: Partial<MemoryMetadata> = {}) {\n    // SECURITY FIX: Sanitize all inputs during construction\n    const sanitizedMetadata = {\n      ...metadata,\n      name: metadata.name ? \n        sanitizeInput(UnicodeValidator.normalize(metadata.name).normalizedContent, 100) : \n        'Unnamed Memory',\n      description: metadata.description ? \n        sanitizeInput(UnicodeValidator.normalize(metadata.description).normalizedContent, 500) : \n        undefined\n    };\n    \n    super(ElementType.MEMORY, sanitizedMetadata);\n    \n    // Initialize memory-specific properties with defaults\n    this.storageBackend = metadata.storageBackend || MEMORY_CONSTANTS.DEFAULT_STORAGE_BACKEND;\n    this.retentionDays = metadata.retentionDays || MEMORY_CONSTANTS.DEFAULT_RETENTION_DAYS;\n    // Validate privacy level - default to private if invalid\n    this.privacyLevel = (metadata.privacyLevel && MEMORY_CONSTANTS.PRIVACY_LEVELS.includes(metadata.privacyLevel)) \n      ? metadata.privacyLevel \n      : MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL;\n    this.searchable = metadata.searchable !== false;\n    this.maxEntries = Math.min(\n      metadata.maxEntries || MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT,\n      MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT\n    );\n    \n    // Set up extensions\n    this.extensions = {\n      storageBackend: this.storageBackend,\n      retentionDays: this.retentionDays,\n      privacyLevel: this.privacyLevel,\n      searchable: this.searchable,\n      maxEntries: this.maxEntries,\n      encryptionEnabled: metadata.encryptionEnabled || false\n    };\n\n    // Initialize search index with configuration (Issue #984)\n    const indexConfig: SearchIndexConfig = {\n      indexThreshold: metadata.indexThreshold || 100,\n      enableContentIndex: metadata.enableContentIndex !== false,\n      maxTermsPerEntry: metadata.maxTermsPerEntry || 100,\n      minTermLength: metadata.minTermLength || 2,\n      enablePersistence: false // Future enhancement\n    };\n    this.searchIndex = new MemorySearchIndex(indexConfig);\n\n    // Log memory creation\n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_CREATED,\n      severity: 'LOW',\n      source: 'Memory.constructor',\n      details: `Memory created: ${this.metadata.name} with ${this.storageBackend} backend`\n    });\n  }\n  \n  /**\n   * Add a new memory entry\n   * SECURITY: Validates and sanitizes all input, enforces size limits\n   */\n  public async addEntry(content: string, tags?: string[], metadata?: Record<string, any>): Promise<MemoryEntry> {\n    // Validate memory size limits\n    if (this.entries.size >= this.maxEntries) {\n      // SECURITY FIX: Enforce retention policy when at capacity\n      await this.enforceRetentionPolicy();\n      \n      // If still at capacity after retention, remove oldest to make room\n      if (this.entries.size >= this.maxEntries) {\n        const oldestEntry = Array.from(this.entries.values())\n          .sort((a, b) => a.timestamp.getTime() - b.timestamp.getTime())[0];\n        if (oldestEntry) {\n          this.entries.delete(oldestEntry.id);\n        }\n      }\n    }\n    \n    // SECURITY FIX: Validate and sanitize content\n    const sanitizedContent = sanitizeMemoryContent(content, MEMORY_CONSTANTS.MAX_ENTRY_SIZE);\n    \n    if (!sanitizedContent || sanitizedContent.trim().length === 0) {\n      throw new Error('Memory content cannot be empty');\n    }\n    \n    // SECURITY FIX: Validate and sanitize tags\n    const sanitizedTags = tags ? this.sanitizeTags(tags) : [];\n    \n    // Create memory entry with generated ID\n    const entry: MemoryEntry = {\n      id: generateMemoryId(),\n      timestamp: new Date(),\n      content: sanitizedContent,\n      tags: sanitizedTags,\n      metadata: this.sanitizeMetadata(metadata),\n      privacyLevel: this.privacyLevel,\n      expiresAt: this.calculateExpiryDate()\n    };\n    \n    // Store entry\n    this.entries.set(entry.id, entry);\n    this._isDirty = true;\n\n    // Update search index (Issue #984)\n    this.searchIndex.addEntry(entry);\n\n    // Check if we should build/rebuild the index\n    if (!this.searchIndex.isIndexed && this.entries.size >= 100) {\n      // Build index asynchronously to avoid blocking, with retry logic\n      this.buildSearchIndexWithRetry().catch(error => {\n        // Final failure after retries - search will fall back to linear scan\n        logger.error('Failed to build search index after retries, search will use fallback', error);\n      });\n    }\n\n    // Log memory addition\n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_ADDED,\n      severity: 'LOW',\n      source: 'Memory.addEntry',\n      details: `Added memory entry ${entry.id} with ${sanitizedTags.length} tags`\n    });\n\n    return entry;\n  }\n  \n  /**\n   * Search memory entries\n   * SECURITY: Respects privacy levels and sanitizes search queries\n   *\n   * IMPLEMENTED: Basic indexed search for O(log n) performance (Issue #984)\n   * - Tag index: Map<tag, Set<entryId>> for instant tag lookups ✓\n   * - Content index: Inverted index for term search ✓\n   * - Date index: Binary tree for efficient range queries ✓\n   * - Privacy index: Pre-sorted entries by privacy level ✓\n   *\n   * TODO: Advanced indexing features (Future enhancements):\n   * - Composite indices: Combined indices for common query patterns\n   * - Index-of-indexes pattern:\n   *   - Master index file (meta.yaml) with pointers to shard indices\n   *   - Each shard maintains its own local index\n   *   - Periodic index compaction and optimization\n   * - Persistent index storage to disk\n   * - Incremental index updates for large datasets\n   *\n   * Performance improvement achieved:\n   * - Previous: ~100ms for 10,000 entries (linear scan)\n   * - Current: <5ms for same dataset (indexed search)\n   */\n  public async search(options: MemorySearchOptions = {}): Promise<MemoryEntry[]> {\n    // SECURITY FIX: Sanitize search query (use regular sanitizeInput for queries)\n    const sanitizedQuery = options.query ?\n      sanitizeInput(UnicodeValidator.normalize(options.query).normalizedContent, 200) :\n      undefined;\n\n    // Use indexed search if available (Issue #984)\n    if (this.searchIndex.isIndexed) {\n      const searchQuery: SearchQuery = {\n        content: sanitizedQuery,\n        tags: options.tags ? this.sanitizeTags(options.tags) : undefined,\n        dateFrom: options.startDate,\n        dateTo: options.endDate,\n        privacyLevel: options.privacyLevel as PrivacyLevel,\n        limit: options.limit\n      };\n\n      const searchResults = this.searchIndex.search(searchQuery, this.entries);\n      return searchResults.map(result => result.entry);\n    }\n\n    // Fallback to linear search for small datasets\n    let results: MemoryEntry[] = [];\n    const queryLower = sanitizedQuery?.toLowerCase();\n    const searchTags = options.tags && options.tags.length > 0 ? this.sanitizeTags(options.tags) : null;\n\n    // Single iteration through entries with all filters applied\n    for (const entry of this.entries.values()) {\n      // Privacy level check\n      if (options.privacyLevel && \n          !this.canAccessPrivacyLevel(entry.privacyLevel || MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL, options.privacyLevel)) {\n        continue;\n      }\n      \n      // Query text check\n      if (queryLower) {\n        const contentMatch = entry.content.toLowerCase().includes(queryLower);\n        const tagMatch = entry.tags?.some(tag => tag.toLowerCase().includes(queryLower));\n        if (!contentMatch && !tagMatch) {\n          continue;\n        }\n      }\n      \n      // Tag filter check\n      if (searchTags && !searchTags.some(searchTag => entry.tags?.includes(searchTag))) {\n        continue;\n      }\n      \n      // Date range checks\n      if (options.startDate && entry.timestamp < options.startDate) {\n        continue;\n      }\n      if (options.endDate && entry.timestamp > options.endDate) {\n        continue;\n      }\n      \n      // Entry passes all filters\n      results.push(entry);\n    }\n    \n    // Sort by timestamp (newest first) - using string comparison for IDs as secondary sort\n    results.sort((a, b) => {\n      const timeDiff = b.timestamp.getTime() - a.timestamp.getTime();\n      if (timeDiff !== 0) return timeDiff;\n      // If timestamps are exactly the same, sort by ID (which contains timestamp)\n      return b.id.localeCompare(a.id);\n    });\n    \n    // Apply limit\n    if (options.limit && options.limit > 0) {\n      results = results.slice(0, options.limit);\n    }\n    \n    // Log search operation\n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_SEARCHED,\n      severity: 'LOW',\n      source: 'Memory.search',\n      details: `Searched memories with query: ${sanitizedQuery || 'none'}, found ${results.length} results`\n    });\n    \n    return results;\n  }\n  \n  /**\n   * Get a specific memory entry by ID\n   */\n  public async getEntry(id: string): Promise<MemoryEntry | undefined> {\n    return this.entries.get(id);\n  }\n  \n  /**\n   * Delete a memory entry\n   * SECURITY: Validates permissions and logs deletion\n   */\n  public async deleteEntry(id: string): Promise<boolean> {\n    const entry = this.entries.get(id);\n    if (!entry) {\n      return false;\n    }\n    \n    // SECURITY: Check if sensitive memories can be deleted\n    if (entry.privacyLevel === 'sensitive') {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.SENSITIVE_MEMORY_DELETED,\n        severity: 'MEDIUM',\n        source: 'Memory.deleteEntry',\n        details: `Sensitive memory ${id} deleted`\n      });\n    }\n    \n    this.entries.delete(id);\n    this._isDirty = true;\n    \n    return true;\n  }\n  \n  /**\n   * Enforce retention policy by removing expired entries\n   * SECURITY: Ensures memory doesn't grow unbounded\n   */\n  public async enforceRetentionPolicy(): Promise<number> {\n    const now = new Date();\n    let deletedCount = 0;\n    \n    // Remove expired entries\n    for (const [id, entry] of this.entries) {\n      if (entry.expiresAt && entry.expiresAt < now) {\n        this.entries.delete(id);\n        deletedCount++;\n      }\n    }\n    \n    // If still at or over capacity, remove oldest entries to make room for one more\n    if (this.entries.size >= this.maxEntries) {\n      const sortedEntries = Array.from(this.entries.entries())\n        .sort((a, b) => a[1].timestamp.getTime() - b[1].timestamp.getTime());\n      \n      // Remove one extra to make room for new entry\n      const toDelete = Math.max(1, this.entries.size - this.maxEntries + 1);\n      for (let i = 0; i < toDelete && i < sortedEntries.length; i++) {\n        this.entries.delete(sortedEntries[i][0]);\n        deletedCount++;\n      }\n    }\n    \n    if (deletedCount > 0) {\n      this._isDirty = true;\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.RETENTION_POLICY_ENFORCED,\n        severity: 'LOW',\n        source: 'Memory.enforceRetentionPolicy',\n        details: `Removed ${deletedCount} expired memories`\n      });\n    }\n    \n    return deletedCount;\n  }\n  \n  /**\n   * Clear all memory entries\n   * SECURITY: Requires confirmation and logs the action\n   */\n  public async clearAll(confirm: boolean = false): Promise<void> {\n    if (!confirm) {\n      throw new Error('Memory clear requires confirmation');\n    }\n    \n    const count = this.entries.size;\n    this.entries.clear();\n    this._isDirty = true;\n    \n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_CLEARED,\n      severity: 'HIGH',\n      source: 'Memory.clearAll',\n      details: `Cleared all ${count} memory entries`\n    });\n  }\n  \n  /**\n   * Get memory statistics\n   */\n  public getStats(): {\n    totalEntries: number;\n    totalSize: number;\n    oldestEntry?: Date;\n    newestEntry?: Date;\n    tagFrequency: Map<string, number>;\n  } {\n    let totalSize = 0;\n    let oldestEntry: Date | undefined;\n    let newestEntry: Date | undefined;\n    const tagFrequency = new Map<string, number>();\n    \n    for (const entry of this.entries.values()) {\n      totalSize += entry.content.length;\n      \n      if (!oldestEntry || entry.timestamp < oldestEntry) {\n        oldestEntry = entry.timestamp;\n      }\n      if (!newestEntry || entry.timestamp > newestEntry) {\n        newestEntry = entry.timestamp;\n      }\n      \n      entry.tags?.forEach(tag => {\n        tagFrequency.set(tag, (tagFrequency.get(tag) || 0) + 1);\n      });\n    }\n    \n    return {\n      totalEntries: this.entries.size,\n      totalSize,\n      oldestEntry,\n      newestEntry,\n      tagFrequency\n    };\n  }\n  \n  /**\n   * Validate the memory element\n   */\n  public override validate(): ElementValidationResult {\n    const result = super.validate();\n    \n    // Initialize errors array if not present\n    if (!result.errors) {\n      result.errors = [];\n    }\n    \n    // Additional memory-specific validation\n    if (this.retentionDays < MEMORY_CONSTANTS.MIN_RETENTION_DAYS || this.retentionDays > MEMORY_CONSTANTS.MAX_RETENTION_DAYS) {\n      result.errors.push({\n        field: 'retentionDays',\n        message: `Retention days must be between ${MEMORY_CONSTANTS.MIN_RETENTION_DAYS} and ${MEMORY_CONSTANTS.MAX_RETENTION_DAYS}`,\n        severity: 'error'\n      } as ValidationError);\n    }\n    \n    if (this.maxEntries < 1 || this.maxEntries > MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT) {\n      result.errors.push({\n        field: 'maxEntries',\n        message: `Max entries must be between 1 and ${MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT}`,\n        severity: 'error'\n      } as ValidationError);\n    }\n    \n    // Check memory size\n    const stats = this.getStats();\n    if (stats.totalSize > MEMORY_CONSTANTS.MAX_MEMORY_SIZE) {\n      result.errors.push({\n        field: 'memory',\n        message: `Total memory size (${stats.totalSize}) exceeds limit (${MEMORY_CONSTANTS.MAX_MEMORY_SIZE})`,\n        severity: 'error'\n      } as ValidationError);\n    }\n    \n    // Update validity based on our checks\n    return {\n      ...result,\n      valid: result.errors.length === 0\n    };\n  }\n  \n  /**\n   * Serialize memory to string\n   */\n  public override serialize(): string {\n    const data = {\n      id: this.id,\n      type: this.type,\n      version: this.version,\n      metadata: this.metadata,\n      extensions: this.extensions,\n      entries: Array.from(this.entries.values())\n    };\n    \n    return JSON.stringify(data, null, 2);\n  }\n  \n  /**\n   * Deserialize memory from string\n   * SECURITY: Validates all loaded data\n   */\n  public override deserialize(data: string): void {\n    try {\n      const parsed = JSON.parse(data);\n      \n      // Validate basic structure\n      if (!parsed.id || !parsed.type || parsed.type !== ElementType.MEMORY) {\n        throw new Error('Invalid memory data format');\n      }\n      \n      // Update properties\n      this.id = parsed.id;\n      this.version = parsed.version || '1.0.0';\n      this.metadata = parsed.metadata || {};\n      this.extensions = parsed.extensions || {};\n      \n      // Clear and reload entries\n      this.entries.clear();\n      if (Array.isArray(parsed.entries)) {\n        for (const entry of parsed.entries) {\n          if (this.isValidEntry(entry)) {\n            // Use optimized sanitization with caching\n            entry.content = this.sanitizeWithCache(entry.content, MEMORY_CONSTANTS.MAX_ENTRY_SIZE);\n            entry.tags = this.sanitizeTags(entry.tags || []);\n            entry.timestamp = new Date(entry.timestamp);\n            if (entry.expiresAt) {\n              entry.expiresAt = new Date(entry.expiresAt);\n            }\n            this.entries.set(entry.id, entry);\n          }\n        }\n      }\n      \n      // Enforce retention policy after loading\n      this.enforceRetentionPolicy();\n      \n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_DESERIALIZE_FAILED,\n        severity: 'HIGH',\n        source: 'Memory.deserialize',\n        details: `Failed to deserialize memory: ${error}`\n      });\n      throw new Error(`Failed to deserialize memory: ${error}`);\n    }\n  }\n  \n  // Private helper methods\n  \n  private calculateExpiryDate(): Date {\n    const expiry = new Date();\n    expiry.setDate(expiry.getDate() + this.retentionDays);\n    return expiry;\n  }\n  \n  private sanitizeTags(tags: string[]): string[] {\n    // SECURITY FIX: Limit number of tags and sanitize each\n    const limitedTags = tags.slice(0, MEMORY_CONSTANTS.MAX_TAGS_PER_ENTRY);\n    \n    return limitedTags\n      .map(tag => {\n        const normalized = UnicodeValidator.normalize(tag).normalizedContent;\n        return sanitizeInput(normalized, MEMORY_CONSTANTS.MAX_TAG_LENGTH);\n      })\n      .filter(tag => tag && tag.length > 0);\n  }\n  \n  private sanitizeMetadata(metadata?: Record<string, any>): Record<string, any> | undefined {\n    if (!metadata) return undefined;\n    \n    // SECURITY FIX: Sanitize metadata values\n    const sanitized: Record<string, any> = {};\n    const maxKeys = MEMORY_CONSTANTS.MAX_METADATA_KEYS;\n    let keyCount = 0;\n    \n    for (const [key, value] of Object.entries(metadata)) {\n      if (keyCount >= maxKeys) break;\n      \n      const sanitizedKey = sanitizeInput(key, MEMORY_CONSTANTS.MAX_METADATA_KEY_LENGTH);\n      if (sanitizedKey && typeof value === 'string') {\n        sanitized[sanitizedKey] = sanitizeInput(value, MEMORY_CONSTANTS.MAX_METADATA_VALUE_LENGTH);\n        keyCount++;\n      } else if (sanitizedKey && typeof value === 'number') {\n        sanitized[sanitizedKey] = value;\n        keyCount++;\n      }\n      // Skip other types for security\n    }\n    \n    return sanitized;\n  }\n  \n  private canAccessPrivacyLevel(entryLevel: string, requestedLevel: string): boolean {\n    const levels = MEMORY_CONSTANTS.PRIVACY_LEVELS;\n    const entryIndex = levels.indexOf(entryLevel as PrivacyLevel);\n    const requestedIndex = levels.indexOf(requestedLevel as PrivacyLevel);\n    \n    // Can only access entries at or below the requested privacy level\n    // e.g., if requesting 'private', can see 'public' and 'private' but not 'sensitive'\n    return entryIndex <= requestedIndex;\n  }\n  \n  private isValidEntry(entry: any): boolean {\n    return entry &&\n      typeof entry.id === 'string' &&\n      typeof entry.content === 'string' &&\n      entry.timestamp &&\n      (!entry.tags || Array.isArray(entry.tags));\n  }\n\n  /**\n   * Optimized sanitization with checksum caching\n   * Avoids re-sanitizing content that hasn't changed\n   * @param content Content to sanitize\n   * @param maxLength Maximum allowed length\n   * @returns Sanitized content\n   */\n  private sanitizeWithCache(content: string, maxLength: number): string {\n    // Generate checksum for the input\n    const checksum = crypto.createHash('sha256').update(content).digest('hex');\n\n    // Check if we've already sanitized this exact content\n    const cacheKey = `${checksum}:${maxLength}`;\n    const cached = this.sanitizationCache.get(cacheKey);\n    if (cached) {\n      return cached;\n    }\n\n    // Perform sanitization\n    const sanitized = sanitizeMemoryContent(content, maxLength);\n\n    // Cache the result (limit cache size to prevent memory issues)\n    if (this.sanitizationCache.size > 1000) {\n      // Remove oldest entries (simple FIFO)\n      const firstKey = this.sanitizationCache.keys().next().value;\n      if (firstKey) this.sanitizationCache.delete(firstKey);\n    }\n    this.sanitizationCache.set(cacheKey, sanitized);\n\n    return sanitized;\n  }\n\n  /**\n   * Build search index with retry logic\n   * Attempts to build the index up to 3 times with exponential backoff\n   * This ensures search functionality even if there are transient failures\n   */\n  private async buildSearchIndexWithRetry(retries = 3): Promise<void> {\n    let lastError: Error | undefined;\n\n    for (let attempt = 1; attempt <= retries; attempt++) {\n      try {\n        await this.searchIndex.buildIndex(this.entries);\n        logger.debug(`Search index built successfully on attempt ${attempt}`);\n        return;\n      } catch (error) {\n        lastError = error instanceof Error ? error : new Error(String(error));\n        logger.warn(`Search index build attempt ${attempt} failed`, {\n          error: lastError.message,\n          entriesCount: this.entries.size\n        });\n\n        if (attempt < retries) {\n          // Exponential backoff: 100ms, 200ms, 400ms\n          const delay = Math.pow(2, attempt - 1) * 100;\n          await new Promise(resolve => setTimeout(resolve, delay));\n        }\n      }\n    }\n\n    // If we get here, all retries failed\n    throw lastError || new Error('Failed to build search index');\n  }\n}"]}
644
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Memory.js","sourceRoot":"","sources":["../../../src/elements/memories/Memory.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Memory Element - Persistent context storage for continuity and learning\n * \n * Provides multiple storage backends, retention policies, and search capabilities\n * for maintaining context across sessions and interactions.\n * \n * SECURITY MEASURES IMPLEMENTED:\n * 1. Input sanitization for all memory content\n * 2. Memory size limits to prevent unbounded growth\n * 3. Path validation for file-based storage\n * 4. Retention policy enforcement\n * 5. Privacy level access control\n * 6. Audit logging for all operations\n */\n\nimport { BaseElement } from '../BaseElement.js';\nimport { IElement, ElementValidationResult, ValidationError } from '../../types/elements/index.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { IElementMetadata } from '../../types/elements/IElement.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport crypto from 'crypto';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { sanitizeInput } from '../../security/InputValidator.js';\nimport { MEMORY_CONSTANTS, MEMORY_SECURITY_EVENTS, PrivacyLevel, StorageBackend } from './constants.js';\nimport { generateMemoryId } from './utils.js';\nimport { MemorySearchIndex, SearchQuery, SearchIndexConfig } from './MemorySearchIndex.js';\nimport { logger } from '../../utils/logger.js';\nimport DOMPurify from 'dompurify';\nimport { JSDOM } from 'jsdom';\n\n// Initialize DOMPurify with JSDOM\nconst window = new JSDOM('').window;\nconst purify = DOMPurify(window as any);\n\n// Configure DOMPurify for memory content - strip all HTML but keep text\npurify.setConfig({\n  ALLOWED_TAGS: [],  // No HTML tags allowed\n  ALLOWED_ATTR: [],  // No attributes allowed\n  KEEP_CONTENT: true // Keep text content\n});\n\n/**\n * Sanitize content for memory storage\n * More permissive than sanitizeInput - allows punctuation, quotes, etc.\n * but still prevents XSS and control characters\n */\nfunction sanitizeMemoryContent(content: string, maxLength: number): string {\n  if (!content || typeof content !== 'string') {\n    return '';\n  }\n  \n  // First normalize Unicode\n  const normalized = UnicodeValidator.normalize(content).normalizedContent;\n  \n  // Use DOMPurify to strip any HTML/XSS attempts but keep text\n  const cleaned = purify.sanitize(normalized);\n  \n  // Remove only control characters and null bytes\n  return cleaned\n    .replace(/[\\x00-\\x08\\x0B\\x0C\\x0E-\\x1F\\x7F]/g, '') // Remove control chars except \\t \\n \\r\n    .substring(0, maxLength)\n    .trim();\n}\n\nexport interface MemoryMetadata extends IElementMetadata {\n  storageBackend?: StorageBackend;\n  retentionDays?: number;\n  privacyLevel?: PrivacyLevel;\n  searchable?: boolean;\n  maxEntries?: number;\n  encryptionEnabled?: boolean;\n  // Search index configuration (Issue #984)\n  indexThreshold?: number;\n  enableContentIndex?: boolean;\n  maxTermsPerEntry?: number;\n  minTermLength?: number;\n}\n\nexport interface MemoryEntry {\n  id: string;\n  timestamp: Date;\n  content: string;\n  tags?: string[];\n  metadata?: Record<string, any>;\n  expiresAt?: Date;\n  privacyLevel?: PrivacyLevel;\n}\n\nexport interface MemorySearchOptions {\n  query?: string;\n  tags?: string[];\n  startDate?: Date;\n  endDate?: Date;\n  limit?: number;\n  privacyLevel?: PrivacyLevel;\n}\n\n/**\n * Memory Element Implementation\n *\n * TODO: Memory Sharding Strategy (Issue #981)\n * ---------------------------------------------\n * Current: Single Map<id, entry> for all memories\n * Problem: Large memory sets (>10K entries) cause performance degradation\n *\n * Planned Sharding Architecture:\n * 1. Shard memories across multiple files based on hash(memoryId) % shardCount\n * 2. Each shard file <256KB for optimal YAML parsing\n * 3. Memory references stored separately from content (like git objects)\n * 4. Large binary content (PDFs, images) stored as external references\n *\n * Benefits:\n * - Parallel loading of memory shards\n * - Reduced memory footprint (load only needed shards)\n * - Better corruption resistance (one shard failure doesn't affect others)\n * - Efficient incremental updates\n *\n * TODO: Content Integrity Verification (Issue #982)\n * --------------------------------------------------\n * Add SHA-256 hashes to detect:\n * - Accidental corruption from disk errors\n * - Intentional tampering with memory files\n * - Version conflicts during concurrent access\n *\n * Implementation:\n * - Store hash in memory metadata\n * - Verify on load, warn on mismatch\n * - Option to auto-restore from backup on corruption\n *\n * TODO: Memory Capacity Management (Issue #983)\n * ---------------------------------------------\n * Current: Synchronous retention enforcement on each add\n * Better: Background cleanup with smart triggers:\n * - Cleanup when 90% capacity reached\n * - Batch deletions for efficiency\n * - LRU eviction with access tracking\n * - Preserve \"pinned\" memories regardless of age\n */\nexport class Memory extends BaseElement implements IElement {\n  // Memory-specific properties\n  private entries: Map<string, MemoryEntry> = new Map();\n  private storageBackend: StorageBackend;\n  private retentionDays: number;\n  private privacyLevel: PrivacyLevel;\n  private searchable: boolean;\n  private maxEntries: number;\n\n  // Search index for performance (Issue #984)\n  private searchIndex: MemorySearchIndex;\n\n  // Sanitization cache to avoid redundant processing\n  private sanitizationCache: Map<string, string> = new Map();\n  \n  constructor(metadata: Partial<MemoryMetadata> = {}) {\n    // SECURITY FIX: Sanitize all inputs during construction\n    const sanitizedMetadata = {\n      ...metadata,\n      name: metadata.name ? \n        sanitizeInput(UnicodeValidator.normalize(metadata.name).normalizedContent, 100) : \n        'Unnamed Memory',\n      description: metadata.description ? \n        sanitizeInput(UnicodeValidator.normalize(metadata.description).normalizedContent, 500) : \n        undefined\n    };\n    \n    super(ElementType.MEMORY, sanitizedMetadata);\n    \n    // Initialize memory-specific properties with defaults\n    this.storageBackend = metadata.storageBackend || MEMORY_CONSTANTS.DEFAULT_STORAGE_BACKEND;\n    this.retentionDays = metadata.retentionDays || MEMORY_CONSTANTS.DEFAULT_RETENTION_DAYS;\n    // Validate privacy level - default to private if invalid\n    this.privacyLevel = (metadata.privacyLevel && MEMORY_CONSTANTS.PRIVACY_LEVELS.includes(metadata.privacyLevel)) \n      ? metadata.privacyLevel \n      : MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL;\n    this.searchable = metadata.searchable !== false;\n    this.maxEntries = Math.min(\n      metadata.maxEntries || MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT,\n      MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT\n    );\n    \n    // Set up extensions\n    this.extensions = {\n      storageBackend: this.storageBackend,\n      retentionDays: this.retentionDays,\n      privacyLevel: this.privacyLevel,\n      searchable: this.searchable,\n      maxEntries: this.maxEntries,\n      encryptionEnabled: metadata.encryptionEnabled || false\n    };\n\n    // Initialize search index with configuration (Issue #984)\n    const indexConfig: SearchIndexConfig = {\n      indexThreshold: metadata.indexThreshold || 100,\n      enableContentIndex: metadata.enableContentIndex !== false,\n      maxTermsPerEntry: metadata.maxTermsPerEntry || 100,\n      minTermLength: metadata.minTermLength || 2,\n      enablePersistence: false // Future enhancement\n    };\n    this.searchIndex = new MemorySearchIndex(indexConfig);\n\n    // Log memory creation\n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_CREATED,\n      severity: 'LOW',\n      source: 'Memory.constructor',\n      details: `Memory created: ${this.metadata.name} with ${this.storageBackend} backend`\n    });\n  }\n  \n  /**\n   * Add a new memory entry\n   * SECURITY: Validates and sanitizes all input, enforces size limits\n   */\n  public async addEntry(content: string, tags?: string[], metadata?: Record<string, any>): Promise<MemoryEntry> {\n    // Validate memory size limits\n    if (this.entries.size >= this.maxEntries) {\n      // SECURITY FIX: Enforce retention policy when at capacity\n      await this.enforceRetentionPolicy();\n      \n      // If still at capacity after retention, remove oldest to make room\n      if (this.entries.size >= this.maxEntries) {\n        const oldestEntry = Array.from(this.entries.values())\n          .sort((a, b) => a.timestamp.getTime() - b.timestamp.getTime())[0];\n        if (oldestEntry) {\n          this.entries.delete(oldestEntry.id);\n        }\n      }\n    }\n    \n    // SECURITY FIX: Validate and sanitize content\n    const sanitizedContent = sanitizeMemoryContent(content, MEMORY_CONSTANTS.MAX_ENTRY_SIZE);\n    \n    if (!sanitizedContent || sanitizedContent.trim().length === 0) {\n      throw new Error('Memory content cannot be empty');\n    }\n    \n    // SECURITY FIX: Validate and sanitize tags\n    const sanitizedTags = tags ? this.sanitizeTags(tags) : [];\n    \n    // Create memory entry with generated ID\n    const entry: MemoryEntry = {\n      id: generateMemoryId(),\n      timestamp: new Date(),\n      content: sanitizedContent,\n      tags: sanitizedTags,\n      metadata: this.sanitizeMetadata(metadata),\n      privacyLevel: this.privacyLevel,\n      expiresAt: this.calculateExpiryDate()\n    };\n    \n    // Store entry\n    this.entries.set(entry.id, entry);\n    this._isDirty = true;\n\n    // Update search index (Issue #984)\n    this.searchIndex.addEntry(entry);\n\n    // Check if we should build/rebuild the index\n    if (!this.searchIndex.isIndexed && this.entries.size >= 100) {\n      // Build index asynchronously to avoid blocking, with retry logic\n      this.buildSearchIndexWithRetry().catch(error => {\n        // Final failure after retries - search will fall back to linear scan\n        logger.error('Failed to build search index after retries, search will use fallback', error);\n      });\n    }\n\n    // Log memory addition\n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_ADDED,\n      severity: 'LOW',\n      source: 'Memory.addEntry',\n      details: `Added memory entry ${entry.id} with ${sanitizedTags.length} tags`\n    });\n\n    return entry;\n  }\n  \n  /**\n   * Search memory entries\n   * SECURITY: Respects privacy levels and sanitizes search queries\n   *\n   * IMPLEMENTED: Basic indexed search for O(log n) performance (Issue #984)\n   * - Tag index: Map<tag, Set<entryId>> for instant tag lookups ✓\n   * - Content index: Inverted index for term search ✓\n   * - Date index: Binary tree for efficient range queries ✓\n   * - Privacy index: Pre-sorted entries by privacy level ✓\n   *\n   * TODO: Advanced indexing features (Future enhancements):\n   * - Composite indices: Combined indices for common query patterns\n   * - Index-of-indexes pattern:\n   *   - Master index file (meta.yaml) with pointers to shard indices\n   *   - Each shard maintains its own local index\n   *   - Periodic index compaction and optimization\n   * - Persistent index storage to disk\n   * - Incremental index updates for large datasets\n   *\n   * Performance improvement achieved:\n   * - Previous: ~100ms for 10,000 entries (linear scan)\n   * - Current: <5ms for same dataset (indexed search)\n   */\n  public async search(options: MemorySearchOptions = {}): Promise<MemoryEntry[]> {\n    // SECURITY FIX: Sanitize search query (use regular sanitizeInput for queries)\n    const sanitizedQuery = options.query ?\n      sanitizeInput(UnicodeValidator.normalize(options.query).normalizedContent, 200) :\n      undefined;\n\n    // Use indexed search if available (Issue #984)\n    if (this.searchIndex.isIndexed) {\n      const searchQuery: SearchQuery = {\n        content: sanitizedQuery,\n        tags: options.tags ? this.sanitizeTags(options.tags) : undefined,\n        dateFrom: options.startDate,\n        dateTo: options.endDate,\n        privacyLevel: options.privacyLevel as PrivacyLevel,\n        limit: options.limit\n      };\n\n      const searchResults = this.searchIndex.search(searchQuery, this.entries);\n      return searchResults.map(result => result.entry);\n    }\n\n    // Fallback to linear search for small datasets\n    let results: MemoryEntry[] = [];\n    const queryLower = sanitizedQuery?.toLowerCase();\n    const searchTags = options.tags && options.tags.length > 0 ? this.sanitizeTags(options.tags) : null;\n\n    // Single iteration through entries with all filters applied\n    for (const entry of this.entries.values()) {\n      // Privacy level check\n      if (options.privacyLevel && \n          !this.canAccessPrivacyLevel(entry.privacyLevel || MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL, options.privacyLevel)) {\n        continue;\n      }\n      \n      // Query text check\n      if (queryLower) {\n        const contentMatch = entry.content.toLowerCase().includes(queryLower);\n        const tagMatch = entry.tags?.some(tag => tag.toLowerCase().includes(queryLower));\n        if (!contentMatch && !tagMatch) {\n          continue;\n        }\n      }\n      \n      // Tag filter check\n      if (searchTags && !searchTags.some(searchTag => entry.tags?.includes(searchTag))) {\n        continue;\n      }\n      \n      // Date range checks\n      if (options.startDate && entry.timestamp < options.startDate) {\n        continue;\n      }\n      if (options.endDate && entry.timestamp > options.endDate) {\n        continue;\n      }\n      \n      // Entry passes all filters\n      results.push(entry);\n    }\n    \n    // Sort by timestamp (newest first) - using string comparison for IDs as secondary sort\n    results.sort((a, b) => {\n      const timeDiff = b.timestamp.getTime() - a.timestamp.getTime();\n      if (timeDiff !== 0) return timeDiff;\n      // If timestamps are exactly the same, sort by ID (which contains timestamp)\n      return b.id.localeCompare(a.id);\n    });\n    \n    // Apply limit\n    if (options.limit && options.limit > 0) {\n      results = results.slice(0, options.limit);\n    }\n    \n    // Log search operation\n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_SEARCHED,\n      severity: 'LOW',\n      source: 'Memory.search',\n      details: `Searched memories with query: ${sanitizedQuery || 'none'}, found ${results.length} results`\n    });\n    \n    return results;\n  }\n  \n  /**\n   * Get a specific memory entry by ID\n   */\n  public async getEntry(id: string): Promise<MemoryEntry | undefined> {\n    return this.entries.get(id);\n  }\n  \n  /**\n   * Delete a memory entry\n   * SECURITY: Validates permissions and logs deletion\n   */\n  public async deleteEntry(id: string): Promise<boolean> {\n    const entry = this.entries.get(id);\n    if (!entry) {\n      return false;\n    }\n    \n    // SECURITY: Check if sensitive memories can be deleted\n    if (entry.privacyLevel === 'sensitive') {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.SENSITIVE_MEMORY_DELETED,\n        severity: 'MEDIUM',\n        source: 'Memory.deleteEntry',\n        details: `Sensitive memory ${id} deleted`\n      });\n    }\n    \n    this.entries.delete(id);\n    this._isDirty = true;\n    \n    return true;\n  }\n  \n  /**\n   * Get formatted content of all memory entries\n   * Returns entries as a readable string for display\n   */\n  get content(): string {\n    if (this.entries.size === 0) {\n      return 'No content stored';\n    }\n\n    // Format entries as readable content (newest first)\n    const sortedEntries = Array.from(this.entries.values())\n      .sort((a, b) => b.timestamp.getTime() - a.timestamp.getTime());\n\n    return sortedEntries.map(entry => {\n      const timestamp = entry.timestamp.toISOString();\n      const tags = entry.tags && entry.tags.length > 0 ? ` [${entry.tags.join(', ')}]` : '';\n      return `[${timestamp}]${tags}: ${entry.content}`;\n    }).join('\\n\\n');\n  }\n\n  /**\n   * Enforce retention policy by removing expired entries\n   * SECURITY: Ensures memory doesn't grow unbounded\n   */\n  public async enforceRetentionPolicy(): Promise<number> {\n    const now = new Date();\n    let deletedCount = 0;\n    \n    // Remove expired entries\n    for (const [id, entry] of this.entries) {\n      if (entry.expiresAt && entry.expiresAt < now) {\n        this.entries.delete(id);\n        deletedCount++;\n      }\n    }\n    \n    // If still at or over capacity, remove oldest entries to make room for one more\n    if (this.entries.size >= this.maxEntries) {\n      const sortedEntries = Array.from(this.entries.entries())\n        .sort((a, b) => a[1].timestamp.getTime() - b[1].timestamp.getTime());\n      \n      // Remove one extra to make room for new entry\n      const toDelete = Math.max(1, this.entries.size - this.maxEntries + 1);\n      for (let i = 0; i < toDelete && i < sortedEntries.length; i++) {\n        this.entries.delete(sortedEntries[i][0]);\n        deletedCount++;\n      }\n    }\n    \n    if (deletedCount > 0) {\n      this._isDirty = true;\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.RETENTION_POLICY_ENFORCED,\n        severity: 'LOW',\n        source: 'Memory.enforceRetentionPolicy',\n        details: `Removed ${deletedCount} expired memories`\n      });\n    }\n    \n    return deletedCount;\n  }\n  \n  /**\n   * Clear all memory entries\n   * SECURITY: Requires confirmation and logs the action\n   */\n  public async clearAll(confirm: boolean = false): Promise<void> {\n    if (!confirm) {\n      throw new Error('Memory clear requires confirmation');\n    }\n    \n    const count = this.entries.size;\n    this.entries.clear();\n    this._isDirty = true;\n    \n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_CLEARED,\n      severity: 'HIGH',\n      source: 'Memory.clearAll',\n      details: `Cleared all ${count} memory entries`\n    });\n  }\n  \n  /**\n   * Get memory statistics\n   */\n  public getStats(): {\n    totalEntries: number;\n    totalSize: number;\n    oldestEntry?: Date;\n    newestEntry?: Date;\n    tagFrequency: Map<string, number>;\n  } {\n    let totalSize = 0;\n    let oldestEntry: Date | undefined;\n    let newestEntry: Date | undefined;\n    const tagFrequency = new Map<string, number>();\n    \n    for (const entry of this.entries.values()) {\n      totalSize += entry.content.length;\n      \n      if (!oldestEntry || entry.timestamp < oldestEntry) {\n        oldestEntry = entry.timestamp;\n      }\n      if (!newestEntry || entry.timestamp > newestEntry) {\n        newestEntry = entry.timestamp;\n      }\n      \n      entry.tags?.forEach(tag => {\n        tagFrequency.set(tag, (tagFrequency.get(tag) || 0) + 1);\n      });\n    }\n    \n    return {\n      totalEntries: this.entries.size,\n      totalSize,\n      oldestEntry,\n      newestEntry,\n      tagFrequency\n    };\n  }\n  \n  /**\n   * Validate the memory element\n   */\n  public override validate(): ElementValidationResult {\n    const result = super.validate();\n    \n    // Initialize errors array if not present\n    if (!result.errors) {\n      result.errors = [];\n    }\n    \n    // Additional memory-specific validation\n    if (this.retentionDays < MEMORY_CONSTANTS.MIN_RETENTION_DAYS || this.retentionDays > MEMORY_CONSTANTS.MAX_RETENTION_DAYS) {\n      result.errors.push({\n        field: 'retentionDays',\n        message: `Retention days must be between ${MEMORY_CONSTANTS.MIN_RETENTION_DAYS} and ${MEMORY_CONSTANTS.MAX_RETENTION_DAYS}`,\n        severity: 'error'\n      } as ValidationError);\n    }\n    \n    if (this.maxEntries < 1 || this.maxEntries > MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT) {\n      result.errors.push({\n        field: 'maxEntries',\n        message: `Max entries must be between 1 and ${MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT}`,\n        severity: 'error'\n      } as ValidationError);\n    }\n    \n    // Check memory size\n    const stats = this.getStats();\n    if (stats.totalSize > MEMORY_CONSTANTS.MAX_MEMORY_SIZE) {\n      result.errors.push({\n        field: 'memory',\n        message: `Total memory size (${stats.totalSize}) exceeds limit (${MEMORY_CONSTANTS.MAX_MEMORY_SIZE})`,\n        severity: 'error'\n      } as ValidationError);\n    }\n    \n    // Update validity based on our checks\n    return {\n      ...result,\n      valid: result.errors.length === 0\n    };\n  }\n  \n  /**\n   * Serialize memory to string\n   */\n  public override serialize(): string {\n    const data = {\n      id: this.id,\n      type: this.type,\n      version: this.version,\n      metadata: this.metadata,\n      extensions: this.extensions,\n      entries: Array.from(this.entries.values())\n    };\n    \n    return JSON.stringify(data, null, 2);\n  }\n  \n  /**\n   * Deserialize memory from string\n   * SECURITY: Validates all loaded data\n   */\n  public override deserialize(data: string): void {\n    try {\n      const parsed = JSON.parse(data);\n      \n      // Validate basic structure\n      if (!parsed.id || !parsed.type || parsed.type !== ElementType.MEMORY) {\n        throw new Error('Invalid memory data format');\n      }\n      \n      // Update properties\n      this.id = parsed.id;\n      this.version = parsed.version || '1.0.0';\n      this.metadata = parsed.metadata || {};\n      this.extensions = parsed.extensions || {};\n      \n      // Clear and reload entries\n      this.entries.clear();\n      if (Array.isArray(parsed.entries)) {\n        for (const entry of parsed.entries) {\n          if (this.isValidEntry(entry)) {\n            // Use optimized sanitization with caching\n            entry.content = this.sanitizeWithCache(entry.content, MEMORY_CONSTANTS.MAX_ENTRY_SIZE);\n            entry.tags = this.sanitizeTags(entry.tags || []);\n            entry.timestamp = new Date(entry.timestamp);\n            if (entry.expiresAt) {\n              entry.expiresAt = new Date(entry.expiresAt);\n            }\n            this.entries.set(entry.id, entry);\n          }\n        }\n      }\n      \n      // Enforce retention policy after loading\n      this.enforceRetentionPolicy();\n      \n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_DESERIALIZE_FAILED,\n        severity: 'HIGH',\n        source: 'Memory.deserialize',\n        details: `Failed to deserialize memory: ${error}`\n      });\n      throw new Error(`Failed to deserialize memory: ${error}`);\n    }\n  }\n  \n  // Private helper methods\n  \n  private calculateExpiryDate(): Date {\n    const expiry = new Date();\n    expiry.setDate(expiry.getDate() + this.retentionDays);\n    return expiry;\n  }\n  \n  private sanitizeTags(tags: string[]): string[] {\n    // SECURITY FIX: Limit number of tags and sanitize each\n    const limitedTags = tags.slice(0, MEMORY_CONSTANTS.MAX_TAGS_PER_ENTRY);\n    \n    return limitedTags\n      .map(tag => {\n        const normalized = UnicodeValidator.normalize(tag).normalizedContent;\n        return sanitizeInput(normalized, MEMORY_CONSTANTS.MAX_TAG_LENGTH);\n      })\n      .filter(tag => tag && tag.length > 0);\n  }\n  \n  private sanitizeMetadata(metadata?: Record<string, any>): Record<string, any> | undefined {\n    if (!metadata) return undefined;\n    \n    // SECURITY FIX: Sanitize metadata values\n    const sanitized: Record<string, any> = {};\n    const maxKeys = MEMORY_CONSTANTS.MAX_METADATA_KEYS;\n    let keyCount = 0;\n    \n    for (const [key, value] of Object.entries(metadata)) {\n      if (keyCount >= maxKeys) break;\n      \n      const sanitizedKey = sanitizeInput(key, MEMORY_CONSTANTS.MAX_METADATA_KEY_LENGTH);\n      if (sanitizedKey && typeof value === 'string') {\n        sanitized[sanitizedKey] = sanitizeInput(value, MEMORY_CONSTANTS.MAX_METADATA_VALUE_LENGTH);\n        keyCount++;\n      } else if (sanitizedKey && typeof value === 'number') {\n        sanitized[sanitizedKey] = value;\n        keyCount++;\n      }\n      // Skip other types for security\n    }\n    \n    return sanitized;\n  }\n  \n  private canAccessPrivacyLevel(entryLevel: string, requestedLevel: string): boolean {\n    const levels = MEMORY_CONSTANTS.PRIVACY_LEVELS;\n    const entryIndex = levels.indexOf(entryLevel as PrivacyLevel);\n    const requestedIndex = levels.indexOf(requestedLevel as PrivacyLevel);\n    \n    // Can only access entries at or below the requested privacy level\n    // e.g., if requesting 'private', can see 'public' and 'private' but not 'sensitive'\n    return entryIndex <= requestedIndex;\n  }\n  \n  private isValidEntry(entry: any): boolean {\n    return entry &&\n      typeof entry.id === 'string' &&\n      typeof entry.content === 'string' &&\n      entry.timestamp &&\n      (!entry.tags || Array.isArray(entry.tags));\n  }\n\n  /**\n   * Optimized sanitization with checksum caching\n   * Avoids re-sanitizing content that hasn't changed\n   * @param content Content to sanitize\n   * @param maxLength Maximum allowed length\n   * @returns Sanitized content\n   */\n  private sanitizeWithCache(content: string, maxLength: number): string {\n    // Generate checksum for the input\n    const checksum = crypto.createHash('sha256').update(content).digest('hex');\n\n    // Check if we've already sanitized this exact content\n    const cacheKey = `${checksum}:${maxLength}`;\n    const cached = this.sanitizationCache.get(cacheKey);\n    if (cached) {\n      return cached;\n    }\n\n    // Perform sanitization\n    const sanitized = sanitizeMemoryContent(content, maxLength);\n\n    // Cache the result (limit cache size to prevent memory issues)\n    if (this.sanitizationCache.size > 1000) {\n      // Remove oldest entries (simple FIFO)\n      const firstKey = this.sanitizationCache.keys().next().value;\n      if (firstKey) this.sanitizationCache.delete(firstKey);\n    }\n    this.sanitizationCache.set(cacheKey, sanitized);\n\n    return sanitized;\n  }\n\n  /**\n   * Build search index with retry logic\n   * Attempts to build the index up to 3 times with exponential backoff\n   * This ensures search functionality even if there are transient failures\n   */\n  private async buildSearchIndexWithRetry(retries = 3): Promise<void> {\n    let lastError: Error | undefined;\n\n    for (let attempt = 1; attempt <= retries; attempt++) {\n      try {\n        await this.searchIndex.buildIndex(this.entries);\n        logger.debug(`Search index built successfully on attempt ${attempt}`);\n        return;\n      } catch (error) {\n        lastError = error instanceof Error ? error : new Error(String(error));\n        logger.warn(`Search index build attempt ${attempt} failed`, {\n          error: lastError.message,\n          entriesCount: this.entries.size\n        });\n\n        if (attempt < retries) {\n          // Exponential backoff: 100ms, 200ms, 400ms\n          const delay = Math.pow(2, attempt - 1) * 100;\n          await new Promise(resolve => setTimeout(resolve, delay));\n        }\n      }\n    }\n\n    // If we get here, all retries failed\n    throw lastError || new Error('Failed to build search index');\n  }\n}"]}
@@ -1 +1 @@
1
- {"version":3,"file":"MemoryManager.d.ts","sourceRoot":"","sources":["../../../src/elements/memories/MemoryManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,OAAO,EAAE,MAAM,EAAE,cAAc,EAAe,MAAM,aAAa,CAAC;AAClE,OAAO,EAAE,eAAe,EAAE,MAAM,yCAAyC,CAAC;AAC1E,OAAO,EAAE,uBAAuB,EAAE,MAAM,kCAAkC,CAAC;AAC3E,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AAavD,qBAAa,aAAc,YAAW,eAAe,CAAC,MAAM,CAAC;IAC3D,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,WAAW,CAAkC;IACrD,OAAO,CAAC,gBAAgB,CAAkC;IAI1D,OAAO,CAAC,gBAAgB,CAAyB;IACjD,OAAO,CAAC,yBAAyB,CAAa;;IAO9C;;;;;;;;;OASG;IACG,IAAI,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IA0H7C;;;;;;OAMG;YACW,kBAAkB;IAyBhC;;;OAGG;IACH,OAAO,CAAC,oBAAoB;IAQ5B;;;;;OAKG;YACW,cAAc;IAkC5B;;;;;;;;;OASG;IACG,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA0F7D;;OAEG;IACG,IAAI,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;IA6D/B;;OAEG;IACG,IAAI,CAAC,SAAS,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,OAAO,GAAG,OAAO,CAAC,MAAM,GAAG,SAAS,CAAC;IAKhF;;OAEG;IACG,QAAQ,CAAC,SAAS,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,OAAO,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAK1E;;;OAGG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA6B7C;;OAEG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAUhD;;OAEG;IACG,MAAM,CAAC,QAAQ,EAAE,OAAO,CAAC,cAAc,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC;IAIhE;;;;;;;;;;OAUG;IACG,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,GAAE,MAAM,GAAG,MAAe,GAAG,OAAO,CAAC,MAAM,CAAC;IA0EpF;;OAEG;IACG,aAAa,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAuBrD;;OAEG;IACH,QAAQ,CAAC,OAAO,EAAE,MAAM,GAAG,uBAAuB;IAIlD;;;OAGG;IACH,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAqBvC;;OAEG;IACH,cAAc,IAAI,WAAW;IAI7B;;OAEG;IACH,gBAAgB,IAAI,MAAM;IAM1B;;;;;;;;OAQG;YACW,sBAAsB;IAgCpC,OAAO,CAAC,eAAe;IAwCvB;;OAEG;IACH,OAAO,CAAC,kBAAkB;CAM3B"}
1
+ {"version":3,"file":"MemoryManager.d.ts","sourceRoot":"","sources":["../../../src/elements/memories/MemoryManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,OAAO,EAAE,MAAM,EAAE,cAAc,EAAe,MAAM,aAAa,CAAC;AAClE,OAAO,EAAE,eAAe,EAAE,MAAM,yCAAyC,CAAC;AAC1E,OAAO,EAAE,uBAAuB,EAAE,MAAM,kCAAkC,CAAC;AAC3E,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AAsBvD,qBAAa,aAAc,YAAW,eAAe,CAAC,MAAM,CAAC;IAC3D,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,WAAW,CAAkC;IACrD,OAAO,CAAC,gBAAgB,CAAkC;IAI1D,OAAO,CAAC,gBAAgB,CAAyB;IACjD,OAAO,CAAC,yBAAyB,CAAa;;IAO9C;;;;;;;;;OASG;IACG,IAAI,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAoI7C;;;;;;OAMG;YACW,kBAAkB;IAyBhC;;;OAGG;IACH,OAAO,CAAC,oBAAoB;IAQ5B;;;;;OAKG;YACW,cAAc;IAkC5B;;;;;;;;;OASG;IACG,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA0F7D;;OAEG;IACG,IAAI,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;IA6D/B;;OAEG;IACG,IAAI,CAAC,SAAS,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,OAAO,GAAG,OAAO,CAAC,MAAM,GAAG,SAAS,CAAC;IAKhF;;OAEG;IACG,QAAQ,CAAC,SAAS,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,OAAO,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAK1E;;;OAGG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA6B7C;;OAEG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAUhD;;OAEG;IACG,MAAM,CAAC,QAAQ,EAAE,OAAO,CAAC,cAAc,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC;IAIhE;;;;;;;;;;OAUG;IACG,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,GAAE,MAAM,GAAG,MAAe,GAAG,OAAO,CAAC,MAAM,CAAC;IA0EpF;;OAEG;IACG,aAAa,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAuBrD;;OAEG;IACH,QAAQ,CAAC,OAAO,EAAE,MAAM,GAAG,uBAAuB;IAIlD;;;OAGG;IACH,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAqBvC;;OAEG;IACH,cAAc,IAAI,WAAW;IAI7B;;OAEG;IACH,gBAAgB,IAAI,MAAM;IAM1B;;;;;;;;OAQG;YACW,sBAAsB;IAuCpC,OAAO,CAAC,eAAe;IAwCvB;;OAEG;IACH,OAAO,CAAC,kBAAkB;CAM3B"}
@@ -21,6 +21,14 @@ import * as path from 'path';
21
21
  import * as fs from 'fs/promises';
22
22
  import * as yaml from 'js-yaml';
23
23
  import * as crypto from 'crypto';
24
+ // Character code constants for whitespace detection
25
+ // Makes the code more readable and maintainable
26
+ const WHITESPACE_CHARS = {
27
+ SPACE: 32,
28
+ TAB: 9,
29
+ NEWLINE: 10,
30
+ CARRIAGE_RETURN: 13
31
+ };
24
32
  export class MemoryManager {
25
33
  portfolioManager;
26
34
  memoriesDir;
@@ -88,8 +96,18 @@ export class MemoryManager {
88
96
  // Check if this is pure YAML (doesn't start with frontmatter markers)
89
97
  let parsed;
90
98
  // Efficient format detection without creating trimmed copy
99
+ // Performance optimization: Use character codes instead of regex for whitespace detection
100
+ // Credit: Jeet Singh (@jeetsingh008) - PR #1035
91
101
  let firstNonWhitespace = 0;
92
- while (firstNonWhitespace < content.length && /\s/.test(content[firstNonWhitespace])) {
102
+ while (firstNonWhitespace < content.length) {
103
+ const charCode = content.charCodeAt(firstNonWhitespace);
104
+ // Check if character is NOT whitespace
105
+ if (charCode !== WHITESPACE_CHARS.SPACE &&
106
+ charCode !== WHITESPACE_CHARS.TAB &&
107
+ charCode !== WHITESPACE_CHARS.NEWLINE &&
108
+ charCode !== WHITESPACE_CHARS.CARRIAGE_RETURN) {
109
+ break;
110
+ }
93
111
  firstNonWhitespace++;
94
112
  }
95
113
  // Handle empty content edge case
@@ -584,9 +602,15 @@ export class MemoryManager {
584
602
  */
585
603
  async validateAndResolvePath(filePath) {
586
604
  // SECURITY FIX: Comprehensive path validation
605
+ // Enhanced validation inspired by Jeet Singh (@jeetsingh008) - PR #1035
606
+ // First normalize the path to resolve any ./ or ../ sequences
587
607
  const normalized = path.normalize(filePath);
588
- // Check for path traversal attempts
589
- if (normalized.includes('..') || path.isAbsolute(normalized)) {
608
+ // Check for path traversal attempts - both in original and normalized
609
+ // Check both the normalized path and the original for any traversal patterns
610
+ if (normalized.includes('..') ||
611
+ filePath.includes('..') ||
612
+ path.isAbsolute(normalized) ||
613
+ path.isAbsolute(filePath)) {
590
614
  SecurityMonitor.logSecurityEvent({
591
615
  type: 'PATH_TRAVERSAL_ATTEMPT',
592
616
  severity: 'HIGH',
@@ -654,4 +678,4 @@ export class MemoryManager {
654
678
  return match ? parseInt(match[1]) : MEMORY_CONSTANTS.DEFAULT_RETENTION_DAYS;
655
679
  }
656
680
  }
657
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MemoryManager.js","sourceRoot":"","sources":["../../../src/elements/memories/MemoryManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MemoryManager - Implementation of IElementManager for Memory elements\n * Handles CRUD operations and lifecycle management for memories implementing IElement\n * \n * FIXES IMPLEMENTED:\n * 1. CRITICAL: Fixed race conditions in file operations by using FileLockManager for atomic reads/writes\n * 2. HIGH: Fixed unvalidated YAML parsing vulnerability by using SecureYamlParser\n * 3. MEDIUM: All user inputs are now validated and sanitized\n * 4. MEDIUM: Audit logging added for security operations\n * 5. MEDIUM: Path validation prevents directory traversal attacks\n */\n\nimport { Memory, MemoryMetadata, MemoryEntry } from './Memory.js';\nimport { IElementManager } from '../../types/elements/IElementManager.js';\nimport { ElementValidationResult } from '../../types/elements/IElement.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { PortfolioManager } from '../../portfolio/PortfolioManager.js';\nimport { FileLockManager } from '../../security/fileLockManager.js';\nimport { SecureYamlParser } from '../../security/secureYamlParser.js';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { sanitizeInput } from '../../security/InputValidator.js';\nimport { MEMORY_CONSTANTS, MEMORY_SECURITY_EVENTS } from './constants.js';\nimport * as path from 'path';\nimport * as fs from 'fs/promises';\nimport * as yaml from 'js-yaml';\nimport * as crypto from 'crypto';\n\nexport class MemoryManager implements IElementManager<Memory> {\n  private portfolioManager: PortfolioManager;\n  private memoriesDir: string;\n  private memoryCache: Map<string, Memory> = new Map();\n  private contentHashIndex: Map<string, string> = new Map();\n\n  // PERFORMANCE IMPROVEMENT: Cache for date folders to avoid directory scanning\n  // Invalidated when new folders are created\n  private dateFoldersCache: string[] | null = null;\n  private dateFoldersCacheTimestamp: number = 0;\n  \n  constructor() {\n    this.portfolioManager = PortfolioManager.getInstance();\n    this.memoriesDir = this.portfolioManager.getElementDir(ElementType.MEMORY);\n  }\n  \n  /**\n   * Load a memory from file\n   * SECURITY FIX #1: Uses FileLockManager.atomicReadFile() instead of fs.readFile()\n   * to prevent race conditions and ensure atomic file operations\n   * @param filePath Path to the memory file to load\n   * @returns Promise resolving to the loaded Memory instance\n   * @throws {Error} When file cannot be found or path validation fails\n   * @throws {Error} When YAML parsing fails or content is malformed\n   * @throws {Error} When memory validation fails after loading\n   */\n  async load(filePath: string): Promise<Memory> {\n    try {\n      let fullPath: string | undefined;\n\n      // Check if it's a relative path (no date folder)\n      if (!filePath.includes(path.sep) || !filePath.match(/^\\d{4}-\\d{2}-\\d{2}/)) {\n        // Search in date folders\n        const dateFolders = await this.getDateFolders();\n        let found = false;\n\n        for (const dateFolder of dateFolders) {\n          const testPath = path.join(this.memoriesDir, dateFolder, filePath);\n          if (await fs.access(testPath).then(() => true).catch(() => false)) {\n            fullPath = testPath;\n            found = true;\n            break;\n          }\n        }\n\n        if (!found) {\n          // Fall back to root directory for backward compatibility during transition\n          fullPath = await this.validateAndResolvePath(filePath);\n        }\n      } else {\n        fullPath = await this.validateAndResolvePath(filePath);\n      }\n\n      // Ensure fullPath is defined\n      if (!fullPath) {\n        throw new Error(`Could not resolve path: ${filePath}`);\n      }\n      \n      // Check cache first\n      const cached = this.memoryCache.get(fullPath);\n      if (cached) {\n        return cached;\n      }\n      \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      // HIGH SEVERITY FIX: Use SecureYamlParser to prevent YAML injection attacks\n      // Previously: Could use unsafe YAML parsing\n      // Now: Uses SecureYamlParser which validates content and prevents malicious patterns\n\n      // Memory files are pure YAML (unlike other elements which are markdown with frontmatter)\n      // Check if this is pure YAML (doesn't start with frontmatter markers)\n      let parsed: any;\n\n      // Efficient format detection without creating trimmed copy\n      let firstNonWhitespace = 0;\n      while (firstNonWhitespace < content.length && /\\s/.test(content[firstNonWhitespace])) {\n        firstNonWhitespace++;\n      }\n\n      // Handle empty content edge case\n      if (firstNonWhitespace === content.length) {\n        // Empty or all whitespace file - create minimal valid structure\n        parsed = { data: {}, content: '' };\n      } else if (!content.startsWith('---', firstNonWhitespace)) {\n        // Pure YAML file - wrap it with frontmatter markers for SecureYamlParser\n        const wrappedContent = `---\\n${content}\\n---\\n`;\n        const parseResult = SecureYamlParser.parse(wrappedContent, {\n          maxYamlSize: MEMORY_CONSTANTS.MAX_YAML_SIZE,\n          validateContent: true\n        });\n        // For pure YAML, the entire content becomes the data, no markdown content\n        parsed = { data: parseResult.data, content: '' };\n      } else {\n        // File with frontmatter (shouldn't happen for memories, but handle it)\n        parsed = SecureYamlParser.parse(content, {\n          maxYamlSize: MEMORY_CONSTANTS.MAX_YAML_SIZE,\n          validateContent: true\n        });\n      }\n\n      // Extract metadata and content\n      const { metadata, content: memoryContent } = this.parseMemoryFile(parsed);\n      \n      // Create memory instance\n      const memory = new Memory(metadata);\n      \n      // Load saved entries if present\n      // Memory files have entries as a top-level key in the YAML\n      const entries = parsed.data?.entries;\n      if (entries) {\n        memory.deserialize(JSON.stringify({\n          id: memory.id,\n          type: memory.type,\n          version: memory.version,\n          metadata: memory.metadata,\n          extensions: memory.extensions,\n          entries: entries\n        }));\n      }\n      \n      // Cache the loaded memory\n      this.memoryCache.set(fullPath, memory);\n      \n      // Log successful load\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_LOADED,\n        severity: 'LOW',\n        source: 'MemoryManager.load',\n        details: `Loaded memory from ${path.basename(fullPath)}`\n      });\n      \n      return memory;\n      \n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_LOAD_FAILED,\n        severity: 'MEDIUM',\n        source: 'MemoryManager.load',\n        details: `Failed to load memory from ${filePath}: ${error}`\n      });\n      throw new Error(`Failed to load memory: ${error}`);\n    }\n  }\n\n  /**\n   * Generate date-based path for memory storage\n   * Creates YYYY-MM-DD folder structure to prevent flat directory issues\n   * @param element Memory element to save\n   * @param fileName Optional custom filename\n   * @returns Full path to memory file\n   */\n  private async generateMemoryPath(element: Memory, fileName?: string): Promise<string> {\n    const date = new Date();\n    const dateFolder = `${date.getFullYear()}-${String(date.getMonth() + 1).padStart(2, '0')}-${String(date.getDate()).padStart(2, '0')}`;\n    const datePath = path.join(this.memoriesDir, dateFolder);\n\n    // Ensure date folder exists\n    await fs.mkdir(datePath, { recursive: true });\n\n    // PERFORMANCE IMPROVEMENT: Invalidate date folders cache since we created a new folder\n    this.dateFoldersCache = null;\n\n    // Generate filename\n    const baseName = fileName || `${element.metadata.name?.toLowerCase().replace(/\\s+/g, '-') || 'memory'}.yaml`;\n    let finalName = baseName;\n    let version = 1;\n\n    // Handle collisions with version suffix\n    while (await fs.access(path.join(datePath, finalName)).then(() => true).catch(() => false)) {\n      version++;\n      finalName = baseName.replace('.yaml', `-v${version}.yaml`);\n    }\n\n    return path.join(datePath, finalName);\n  }\n\n  /**\n   * Calculate SHA-256 hash of memory content for deduplication\n   * Implements Issue #994 - Content-based deduplication\n   */\n  private calculateContentHash(element: Memory): string {\n    const content = JSON.stringify({\n      metadata: element.metadata,\n      entries: JSON.parse(element.serialize()).entries\n    });\n    return crypto.createHash('sha256').update(content).digest('hex');\n  }\n\n  /**\n   * Get all date folders in memories directory\n   * PERFORMANCE IMPROVEMENT: Uses cache to avoid repeated directory scanning\n   * Cache is invalidated when new folders are created or after 60 seconds\n   * @returns Array of date folder names\n   */\n  private async getDateFolders(): Promise<string[]> {\n    const now = Date.now();\n    const CACHE_TTL = 60000; // 60 seconds\n\n    // Return cached result if valid\n    if (this.dateFoldersCache !== null &&\n        (now - this.dateFoldersCacheTimestamp) < CACHE_TTL) {\n      return this.dateFoldersCache;\n    }\n\n    try {\n      const entries = await fs.readdir(this.memoriesDir, { withFileTypes: true });\n      const folders = entries\n        .filter(entry => entry.isDirectory() && /^\\d{4}-\\d{2}-\\d{2}$/.test(entry.name))\n        .map(entry => entry.name)\n        .sort()\n        .reverse(); // Most recent first\n\n      // Cache the result\n      this.dateFoldersCache = folders;\n      this.dateFoldersCacheTimestamp = now;\n\n      return folders;\n    } catch (error) {\n      if ((error as any).code === 'ENOENT') {\n        // Cache empty result\n        this.dateFoldersCache = [];\n        this.dateFoldersCacheTimestamp = now;\n        return [];\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Save a memory to file\n   * SECURITY FIX #1: Uses FileLockManager.atomicWriteFile() for atomic operations\n   * @param element Memory element to save\n   * @param filePath Optional custom file path, defaults to date-based path\n   * @returns Promise that resolves when save is complete\n   * @throws {Error} When memory validation fails before saving\n   * @throws {Error} When path validation fails or file system errors occur\n   * @throws {Error} When atomic write operation fails\n   */\n  async save(element: Memory, filePath?: string): Promise<void> {\n    try {\n      // Validate element\n      const validation = element.validate();\n      if (!validation.valid) {\n        throw new Error(`Invalid memory: ${validation.errors?.map(e => e.message).join(', ')}`);\n      }\n\n      // Calculate content hash for deduplication\n      const contentHash = this.calculateContentHash(element);\n      const existingPath = this.contentHashIndex.get(contentHash);\n\n      if (existingPath) {\n        // Log duplicate detection\n        SecurityMonitor.logSecurityEvent({\n          type: 'MEMORY_DUPLICATE_DETECTED',\n          severity: 'LOW',\n          source: 'MemoryManager.save',\n          details: `Duplicate content detected. Existing: ${existingPath}`\n        });\n      }\n\n      // Generate date-based path if not provided\n      const fullPath = filePath\n        ? await this.validateAndResolvePath(filePath)\n        : await this.generateMemoryPath(element);\n\n      // Ensure parent directory exists (for date folders)\n      await fs.mkdir(path.dirname(fullPath), { recursive: true });\n      \n      // Get memory statistics\n      const stats = element.getStats();\n      \n      // Prepare data for saving\n      const data = {\n        metadata: element.metadata,\n        extensions: element.extensions,\n        stats: {\n          totalEntries: stats.totalEntries,\n          totalSize: stats.totalSize,\n          oldestEntry: stats.oldestEntry?.toISOString(),\n          newestEntry: stats.newestEntry?.toISOString(),\n          topTags: Array.from(stats.tagFrequency.entries())\n            .sort((a, b) => b[1] - a[1])\n            .slice(0, 10)\n            .map(([tag, count]) => ({ tag, count }))\n        },\n        entries: JSON.parse(element.serialize()).entries\n      };\n      \n      // SECURITY FIX: Use secure YAML dumping with safety options\n      // Previously: Could allow dangerous YAML features\n      // Now: Uses FAILSAFE_SCHEMA and security options to prevent code execution\n      const yamlContent = yaml.dump(data, {\n        schema: yaml.FAILSAFE_SCHEMA,\n        noRefs: true,\n        skipInvalid: true,\n        sortKeys: true\n      });\n      \n      // CRITICAL FIX: Use atomic file write to prevent corruption\n      // Previously: await fs.writeFile(fullPath, yamlContent, 'utf-8');\n      // Now: Uses FileLockManager for atomic write with proper encoding\n      await FileLockManager.atomicWriteFile(fullPath, yamlContent, { encoding: 'utf-8' });\n      \n      // Update cache\n      this.memoryCache.set(fullPath, element);\n\n      // Update content hash index\n      this.contentHashIndex.set(contentHash, fullPath);\n\n      // Log successful save\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_SAVED,\n        severity: 'LOW',\n        source: 'MemoryManager.save',\n        details: `Saved memory to ${path.basename(fullPath)} with ${stats.totalEntries} entries`\n      });\n      \n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_SAVE_FAILED,\n        severity: 'HIGH',\n        source: 'MemoryManager.save',\n        details: `Failed to save memory to ${filePath}: ${error}`\n      });\n      throw new Error(`Failed to save memory: ${error}`);\n    }\n  }\n  \n  /**\n   * List all available memories\n   */\n  async list(): Promise<Memory[]> {\n    const memories: Memory[] = [];\n\n    try {\n      // Get all date folders\n      const dateFolders = await this.getDateFolders();\n\n      // Also check root directory for any memory files\n      // Memory files should be .yaml format only\n      const rootFiles = await fs.readdir(this.memoriesDir)\n        .then(files => files.filter(f => f.endsWith('.yaml')))\n        .catch(() => []);\n\n      // Process root files first (legacy)\n      for (const file of rootFiles) {\n        try {\n          const memory = await this.load(file);\n          memories.push(memory);\n        } catch (error) {\n          SecurityMonitor.logSecurityEvent({\n            type: MEMORY_SECURITY_EVENTS.MEMORY_LIST_ITEM_FAILED,\n            severity: 'LOW',\n            source: 'MemoryManager.list',\n            details: `Failed to load ${file}: ${error}`\n          });\n        }\n      }\n\n      // Process date folders\n      for (const dateFolder of dateFolders) {\n        const folderPath = path.join(this.memoriesDir, dateFolder);\n        const files = await fs.readdir(folderPath)\n          .then(files => files.filter(f => f.endsWith('.yaml')))\n          .catch(() => []);\n\n        for (const file of files) {\n          try {\n            const memory = await this.load(path.join(dateFolder, file));\n            memories.push(memory);\n          } catch (error) {\n            SecurityMonitor.logSecurityEvent({\n              type: MEMORY_SECURITY_EVENTS.MEMORY_LIST_ITEM_FAILED,\n              severity: 'LOW',\n              source: 'MemoryManager.list',\n              details: `Failed to load ${dateFolder}/${file}: ${error}`\n            });\n          }\n        }\n      }\n\n      return memories;\n      \n    } catch (error) {\n      if ((error as any).code === 'ENOENT') {\n        // Directory doesn't exist yet\n        return [];\n      }\n      throw error;\n    }\n  }\n  \n  /**\n   * Find memories matching a predicate\n   */\n  async find(predicate: (element: Memory) => boolean): Promise<Memory | undefined> {\n    const memories = await this.list();\n    return memories.find(predicate);\n  }\n  \n  /**\n   * Find multiple memories matching a predicate\n   */\n  async findMany(predicate: (element: Memory) => boolean): Promise<Memory[]> {\n    const memories = await this.list();\n    return memories.filter(predicate);\n  }\n  \n  /**\n   * Delete a memory file\n   * SECURITY: Validates path and logs deletion\n   */\n  async delete(filePath: string): Promise<void> {\n    try {\n      const fullPath = await this.validateAndResolvePath(filePath);\n      \n      // Check if file exists\n      await fs.access(fullPath);\n      \n      // Delete the file\n      await fs.unlink(fullPath);\n      \n      // Remove from cache\n      this.memoryCache.delete(fullPath);\n      \n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_DELETED,\n        severity: 'MEDIUM',\n        source: 'MemoryManager.delete',\n        details: `Deleted memory file: ${path.basename(fullPath)}`\n      });\n      \n    } catch (error) {\n      if ((error as any).code === 'ENOENT') {\n        // File doesn't exist, not an error for delete operation\n        return;\n      }\n      throw error;\n    }\n  }\n  \n  /**\n   * Check if a memory file exists\n   */\n  async exists(filePath: string): Promise<boolean> {\n    try {\n      const fullPath = await this.validateAndResolvePath(filePath);\n      await fs.access(fullPath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Create a new memory with metadata\n   */\n  async create(metadata: Partial<MemoryMetadata>): Promise<Memory> {\n    return new Memory(metadata);\n  }\n  \n  /**\n   * Import a memory from JSON/YAML string\n   * SECURITY: Full validation of imported content\n   * @param data JSON or YAML string containing memory data\n   * @param format Format of the input data ('json' or 'yaml')\n   * @returns Promise resolving to the imported Memory instance\n   * @throws {Error} When JSON/YAML parsing fails\n   * @throws {Error} When imported data is missing required fields\n   * @throws {Error} When YAML content exceeds maximum allowed size\n   * @throws {Error} When imported memory fails validation\n   */\n  async importElement(data: string, format: 'json' | 'yaml' = 'yaml'): Promise<Memory> {\n    try {\n      let parsed: any;\n      \n      if (format === 'json') {\n        parsed = JSON.parse(data);\n      } else {\n        // HIGH SEVERITY FIX: Use secure YAML parsing\n        // Memory import expects pure YAML (not frontmatter), so we parse it securely\n        try {\n          // First validate the YAML content size\n          if (data.length > MEMORY_CONSTANTS.MAX_YAML_SIZE) {\n            throw new Error('YAML content exceeds maximum allowed size');\n          }\n          \n          // Create a wrapper to use SecureYamlParser with pure YAML\n          // Add minimal frontmatter markers to satisfy parser\n          const wrappedYaml = `---\\n${data}\\n---\\n`;\n          \n          const parseResult = SecureYamlParser.parse(wrappedYaml, {\n            maxYamlSize: MEMORY_CONSTANTS.MAX_YAML_SIZE,\n            validateContent: true\n          });\n          \n          // Extract the parsed data (will be in the 'data' property)\n          parsed = parseResult.data;\n          \n        } catch (yamlError) {\n          throw new Error(`Invalid YAML: ${yamlError}`);\n        }\n        \n        // Validate it's an object\n        if (!parsed || typeof parsed !== 'object') {\n          throw new Error('YAML must contain an object');\n        }\n      }\n      \n      // Handle different structures from YAML parsing\n      let metadata = parsed.metadata;\n      let entries = parsed.entries || (parsed.data && parsed.data.entries);\n      \n      // Validate required fields\n      if (!metadata || !metadata.name) {\n        throw new Error('Memory must have metadata with name');\n      }\n      \n      // Create memory instance\n      const memory = new Memory(metadata);\n      \n      // Load entries if present\n      if (entries) {\n        memory.deserialize(JSON.stringify({\n          id: memory.id,\n          type: memory.type,\n          version: memory.version,\n          metadata: memory.metadata,\n          extensions: memory.extensions,\n          entries: entries\n        }));\n      }\n      \n      return memory;\n      \n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_IMPORT_FAILED,\n        severity: 'MEDIUM',\n        source: 'MemoryManager.importElement',\n        details: `Failed to import memory: ${error}`\n      });\n      throw new Error(`Failed to import memory: ${error}`);\n    }\n  }\n  \n  /**\n   * Export a memory to YAML string\n   */\n  async exportElement(element: Memory): Promise<string> {\n    const stats = element.getStats();\n    const data = {\n      metadata: element.metadata,\n      extensions: element.extensions,\n      stats: {\n        totalEntries: stats.totalEntries,\n        totalSize: stats.totalSize,\n        oldestEntry: stats.oldestEntry?.toISOString(),\n        newestEntry: stats.newestEntry?.toISOString()\n      },\n      entries: JSON.parse(element.serialize()).entries\n    };\n    \n    // SECURITY FIX: Use secure YAML dumping\n    return yaml.dump(data, {\n      schema: yaml.FAILSAFE_SCHEMA,\n      noRefs: true,\n      skipInvalid: true,\n      sortKeys: true\n    });\n  }\n  \n  /**\n   * Validate a memory element\n   */\n  validate(element: Memory): ElementValidationResult {\n    return element.validate();\n  }\n  \n  /**\n   * Validate and resolve a file path\n   * SECURITY: Prevents directory traversal attacks\n   */\n  validatePath(filePath: string): boolean {\n    try {\n      // Perform synchronous validation checks\n      const normalized = path.normalize(filePath);\n      \n      // Check for path traversal attempts\n      if (normalized.includes('..') || path.isAbsolute(normalized)) {\n        return false;\n      }\n      \n      // Ensure proper extension\n      if (!normalized.endsWith('.md') && !normalized.endsWith('.yaml') && !normalized.endsWith('.yml')) {\n        return false;\n      }\n      \n      return true;\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Get the element type this manager handles\n   */\n  getElementType(): ElementType {\n    return ElementType.MEMORY;\n  }\n  \n  /**\n   * Get the file extension for memory files\n   */\n  getFileExtension(): string {\n    return '.yaml';\n  }\n  \n  // Private helper methods\n  \n  /**\n   * Validate and resolve a file path to prevent security issues\n   * @param filePath Path to validate and resolve\n   * @returns Promise resolving to the validated full path\n   * @throws {Error} When path contains traversal attempts (../)\n   * @throws {Error} When path is absolute or invalid\n   * @throws {Error} When file extension is not allowed (.md, .yaml, .yml)\n   * @throws {Error} When resolved path would be outside memories directory\n   */\n  private async validateAndResolvePath(filePath: string): Promise<string> {\n    // SECURITY FIX: Comprehensive path validation\n    const normalized = path.normalize(filePath);\n    \n    // Check for path traversal attempts\n    if (normalized.includes('..') || path.isAbsolute(normalized)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'HIGH',\n        source: 'MemoryManager.validateAndResolvePath',\n        details: `Blocked path traversal attempt: ${filePath}`\n      });\n      throw new Error('Invalid file path: Path traversal detected');\n    }\n    \n    // Ensure proper extension - memories should only be .yaml or .yml\n    if (!normalized.endsWith('.yaml') && !normalized.endsWith('.yml')) {\n      throw new Error('Memory files must have .yaml or .yml extension');\n    }\n    \n    // Construct full path\n    const fullPath = path.join(this.memoriesDir, normalized);\n    \n    // Verify it's within memories directory\n    const relative = path.relative(this.memoriesDir, fullPath);\n    if (relative.startsWith('..') || path.isAbsolute(relative)) {\n      throw new Error('File path must be within memories directory');\n    }\n    \n    return fullPath;\n  }\n  \n  private parseMemoryFile(parsed: any): { metadata: MemoryMetadata; content: string } {\n    // FIX: SecureYamlParser returns data in 'data' property, not 'metadata'\n    // For markdown files with YAML frontmatter, the structure is:\n    // parsed.data = YAML frontmatter values\n    // parsed.content = markdown content after frontmatter\n\n    // For pure YAML memory files, we need to check if metadata is directly in data\n    const yamlData = parsed.data || {};\n\n    // Memory files saved by the system have metadata as a top-level key\n    const metadataSource = yamlData.metadata || yamlData;\n\n    // Extract metadata with validation\n    const metadata: MemoryMetadata = {\n      name: sanitizeInput(metadataSource.name || 'Unnamed Memory', 100),\n      description: metadataSource.description ?\n        sanitizeInput(metadataSource.description, 500) :\n        '',\n      version: metadataSource.version || '1.0.0',\n      author: metadataSource.author,\n      created: metadataSource.created,\n      modified: metadataSource.modified || new Date().toISOString(),\n      tags: Array.isArray(metadataSource.tags) ?\n        metadataSource.tags.map((tag: string) => sanitizeInput(tag, MEMORY_CONSTANTS.MAX_TAG_LENGTH)) :\n        [],\n      storageBackend: metadataSource.storage_backend || metadataSource.storageBackend || MEMORY_CONSTANTS.DEFAULT_STORAGE_BACKEND,\n      retentionDays: metadataSource.retention_policy?.default ?\n        this.parseRetentionDays(metadataSource.retention_policy.default) :\n        (metadataSource.retentionDays || MEMORY_CONSTANTS.DEFAULT_RETENTION_DAYS),\n      privacyLevel: metadataSource.privacy_level || metadataSource.privacyLevel || MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL,\n      searchable: metadataSource.searchable !== false,\n      maxEntries: metadataSource.maxEntries || MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT\n    };\n\n    // Extract content (if any)\n    const content = parsed.content || '';\n\n    return { metadata, content };\n  }\n\n  /**\n   * Helper to parse retention days from various formats\n   */\n  private parseRetentionDays(retention: string | number): number {\n    if (typeof retention === 'number') return retention;\n    if (retention === 'permanent' || retention === 'perpetual') return 999999;\n    const match = retention.match(/(\\d+)\\s*days?/i);\n    return match ? parseInt(match[1]) : MEMORY_CONSTANTS.DEFAULT_RETENTION_DAYS;\n  }\n}"]}
681
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MemoryManager.js","sourceRoot":"","sources":["../../../src/elements/memories/MemoryManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MemoryManager - Implementation of IElementManager for Memory elements\n * Handles CRUD operations and lifecycle management for memories implementing IElement\n * \n * FIXES IMPLEMENTED:\n * 1. CRITICAL: Fixed race conditions in file operations by using FileLockManager for atomic reads/writes\n * 2. HIGH: Fixed unvalidated YAML parsing vulnerability by using SecureYamlParser\n * 3. MEDIUM: All user inputs are now validated and sanitized\n * 4. MEDIUM: Audit logging added for security operations\n * 5. MEDIUM: Path validation prevents directory traversal attacks\n */\n\nimport { Memory, MemoryMetadata, MemoryEntry } from './Memory.js';\nimport { IElementManager } from '../../types/elements/IElementManager.js';\nimport { ElementValidationResult } from '../../types/elements/IElement.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { PortfolioManager } from '../../portfolio/PortfolioManager.js';\nimport { FileLockManager } from '../../security/fileLockManager.js';\nimport { SecureYamlParser } from '../../security/secureYamlParser.js';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { sanitizeInput } from '../../security/InputValidator.js';\nimport { MEMORY_CONSTANTS, MEMORY_SECURITY_EVENTS } from './constants.js';\nimport * as path from 'path';\nimport * as fs from 'fs/promises';\nimport * as yaml from 'js-yaml';\nimport * as crypto from 'crypto';\n\n// Character code constants for whitespace detection\n// Makes the code more readable and maintainable\nconst WHITESPACE_CHARS = {\n  SPACE: 32,\n  TAB: 9,\n  NEWLINE: 10,\n  CARRIAGE_RETURN: 13\n} as const;\n\nexport class MemoryManager implements IElementManager<Memory> {\n  private portfolioManager: PortfolioManager;\n  private memoriesDir: string;\n  private memoryCache: Map<string, Memory> = new Map();\n  private contentHashIndex: Map<string, string> = new Map();\n\n  // PERFORMANCE IMPROVEMENT: Cache for date folders to avoid directory scanning\n  // Invalidated when new folders are created\n  private dateFoldersCache: string[] | null = null;\n  private dateFoldersCacheTimestamp: number = 0;\n  \n  constructor() {\n    this.portfolioManager = PortfolioManager.getInstance();\n    this.memoriesDir = this.portfolioManager.getElementDir(ElementType.MEMORY);\n  }\n  \n  /**\n   * Load a memory from file\n   * SECURITY FIX #1: Uses FileLockManager.atomicReadFile() instead of fs.readFile()\n   * to prevent race conditions and ensure atomic file operations\n   * @param filePath Path to the memory file to load\n   * @returns Promise resolving to the loaded Memory instance\n   * @throws {Error} When file cannot be found or path validation fails\n   * @throws {Error} When YAML parsing fails or content is malformed\n   * @throws {Error} When memory validation fails after loading\n   */\n  async load(filePath: string): Promise<Memory> {\n    try {\n      let fullPath: string | undefined;\n\n      // Check if it's a relative path (no date folder)\n      if (!filePath.includes(path.sep) || !filePath.match(/^\\d{4}-\\d{2}-\\d{2}/)) {\n        // Search in date folders\n        const dateFolders = await this.getDateFolders();\n        let found = false;\n\n        for (const dateFolder of dateFolders) {\n          const testPath = path.join(this.memoriesDir, dateFolder, filePath);\n          if (await fs.access(testPath).then(() => true).catch(() => false)) {\n            fullPath = testPath;\n            found = true;\n            break;\n          }\n        }\n\n        if (!found) {\n          // Fall back to root directory for backward compatibility during transition\n          fullPath = await this.validateAndResolvePath(filePath);\n        }\n      } else {\n        fullPath = await this.validateAndResolvePath(filePath);\n      }\n\n      // Ensure fullPath is defined\n      if (!fullPath) {\n        throw new Error(`Could not resolve path: ${filePath}`);\n      }\n      \n      // Check cache first\n      const cached = this.memoryCache.get(fullPath);\n      if (cached) {\n        return cached;\n      }\n      \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      // HIGH SEVERITY FIX: Use SecureYamlParser to prevent YAML injection attacks\n      // Previously: Could use unsafe YAML parsing\n      // Now: Uses SecureYamlParser which validates content and prevents malicious patterns\n\n      // Memory files are pure YAML (unlike other elements which are markdown with frontmatter)\n      // Check if this is pure YAML (doesn't start with frontmatter markers)\n      let parsed: any;\n\n      // Efficient format detection without creating trimmed copy\n      // Performance optimization: Use character codes instead of regex for whitespace detection\n      // Credit: Jeet Singh (@jeetsingh008) - PR #1035\n      let firstNonWhitespace = 0;\n      while (firstNonWhitespace < content.length) {\n        const charCode = content.charCodeAt(firstNonWhitespace);\n        // Check if character is NOT whitespace\n        if (charCode !== WHITESPACE_CHARS.SPACE &&\n            charCode !== WHITESPACE_CHARS.TAB &&\n            charCode !== WHITESPACE_CHARS.NEWLINE &&\n            charCode !== WHITESPACE_CHARS.CARRIAGE_RETURN) {\n          break;\n        }\n        firstNonWhitespace++;\n      }\n\n      // Handle empty content edge case\n      if (firstNonWhitespace === content.length) {\n        // Empty or all whitespace file - create minimal valid structure\n        parsed = { data: {}, content: '' };\n      } else if (!content.startsWith('---', firstNonWhitespace)) {\n        // Pure YAML file - wrap it with frontmatter markers for SecureYamlParser\n        const wrappedContent = `---\\n${content}\\n---\\n`;\n        const parseResult = SecureYamlParser.parse(wrappedContent, {\n          maxYamlSize: MEMORY_CONSTANTS.MAX_YAML_SIZE,\n          validateContent: true\n        });\n        // For pure YAML, the entire content becomes the data, no markdown content\n        parsed = { data: parseResult.data, content: '' };\n      } else {\n        // File with frontmatter (shouldn't happen for memories, but handle it)\n        parsed = SecureYamlParser.parse(content, {\n          maxYamlSize: MEMORY_CONSTANTS.MAX_YAML_SIZE,\n          validateContent: true\n        });\n      }\n\n      // Extract metadata and content\n      const { metadata, content: memoryContent } = this.parseMemoryFile(parsed);\n      \n      // Create memory instance\n      const memory = new Memory(metadata);\n      \n      // Load saved entries if present\n      // Memory files have entries as a top-level key in the YAML\n      const entries = parsed.data?.entries;\n      if (entries) {\n        memory.deserialize(JSON.stringify({\n          id: memory.id,\n          type: memory.type,\n          version: memory.version,\n          metadata: memory.metadata,\n          extensions: memory.extensions,\n          entries: entries\n        }));\n      }\n      \n      // Cache the loaded memory\n      this.memoryCache.set(fullPath, memory);\n      \n      // Log successful load\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_LOADED,\n        severity: 'LOW',\n        source: 'MemoryManager.load',\n        details: `Loaded memory from ${path.basename(fullPath)}`\n      });\n      \n      return memory;\n      \n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_LOAD_FAILED,\n        severity: 'MEDIUM',\n        source: 'MemoryManager.load',\n        details: `Failed to load memory from ${filePath}: ${error}`\n      });\n      throw new Error(`Failed to load memory: ${error}`);\n    }\n  }\n\n  /**\n   * Generate date-based path for memory storage\n   * Creates YYYY-MM-DD folder structure to prevent flat directory issues\n   * @param element Memory element to save\n   * @param fileName Optional custom filename\n   * @returns Full path to memory file\n   */\n  private async generateMemoryPath(element: Memory, fileName?: string): Promise<string> {\n    const date = new Date();\n    const dateFolder = `${date.getFullYear()}-${String(date.getMonth() + 1).padStart(2, '0')}-${String(date.getDate()).padStart(2, '0')}`;\n    const datePath = path.join(this.memoriesDir, dateFolder);\n\n    // Ensure date folder exists\n    await fs.mkdir(datePath, { recursive: true });\n\n    // PERFORMANCE IMPROVEMENT: Invalidate date folders cache since we created a new folder\n    this.dateFoldersCache = null;\n\n    // Generate filename\n    const baseName = fileName || `${element.metadata.name?.toLowerCase().replace(/\\s+/g, '-') || 'memory'}.yaml`;\n    let finalName = baseName;\n    let version = 1;\n\n    // Handle collisions with version suffix\n    while (await fs.access(path.join(datePath, finalName)).then(() => true).catch(() => false)) {\n      version++;\n      finalName = baseName.replace('.yaml', `-v${version}.yaml`);\n    }\n\n    return path.join(datePath, finalName);\n  }\n\n  /**\n   * Calculate SHA-256 hash of memory content for deduplication\n   * Implements Issue #994 - Content-based deduplication\n   */\n  private calculateContentHash(element: Memory): string {\n    const content = JSON.stringify({\n      metadata: element.metadata,\n      entries: JSON.parse(element.serialize()).entries\n    });\n    return crypto.createHash('sha256').update(content).digest('hex');\n  }\n\n  /**\n   * Get all date folders in memories directory\n   * PERFORMANCE IMPROVEMENT: Uses cache to avoid repeated directory scanning\n   * Cache is invalidated when new folders are created or after 60 seconds\n   * @returns Array of date folder names\n   */\n  private async getDateFolders(): Promise<string[]> {\n    const now = Date.now();\n    const CACHE_TTL = 60000; // 60 seconds\n\n    // Return cached result if valid\n    if (this.dateFoldersCache !== null &&\n        (now - this.dateFoldersCacheTimestamp) < CACHE_TTL) {\n      return this.dateFoldersCache;\n    }\n\n    try {\n      const entries = await fs.readdir(this.memoriesDir, { withFileTypes: true });\n      const folders = entries\n        .filter(entry => entry.isDirectory() && /^\\d{4}-\\d{2}-\\d{2}$/.test(entry.name))\n        .map(entry => entry.name)\n        .sort()\n        .reverse(); // Most recent first\n\n      // Cache the result\n      this.dateFoldersCache = folders;\n      this.dateFoldersCacheTimestamp = now;\n\n      return folders;\n    } catch (error) {\n      if ((error as any).code === 'ENOENT') {\n        // Cache empty result\n        this.dateFoldersCache = [];\n        this.dateFoldersCacheTimestamp = now;\n        return [];\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Save a memory to file\n   * SECURITY FIX #1: Uses FileLockManager.atomicWriteFile() for atomic operations\n   * @param element Memory element to save\n   * @param filePath Optional custom file path, defaults to date-based path\n   * @returns Promise that resolves when save is complete\n   * @throws {Error} When memory validation fails before saving\n   * @throws {Error} When path validation fails or file system errors occur\n   * @throws {Error} When atomic write operation fails\n   */\n  async save(element: Memory, filePath?: string): Promise<void> {\n    try {\n      // Validate element\n      const validation = element.validate();\n      if (!validation.valid) {\n        throw new Error(`Invalid memory: ${validation.errors?.map(e => e.message).join(', ')}`);\n      }\n\n      // Calculate content hash for deduplication\n      const contentHash = this.calculateContentHash(element);\n      const existingPath = this.contentHashIndex.get(contentHash);\n\n      if (existingPath) {\n        // Log duplicate detection\n        SecurityMonitor.logSecurityEvent({\n          type: 'MEMORY_DUPLICATE_DETECTED',\n          severity: 'LOW',\n          source: 'MemoryManager.save',\n          details: `Duplicate content detected. Existing: ${existingPath}`\n        });\n      }\n\n      // Generate date-based path if not provided\n      const fullPath = filePath\n        ? await this.validateAndResolvePath(filePath)\n        : await this.generateMemoryPath(element);\n\n      // Ensure parent directory exists (for date folders)\n      await fs.mkdir(path.dirname(fullPath), { recursive: true });\n      \n      // Get memory statistics\n      const stats = element.getStats();\n      \n      // Prepare data for saving\n      const data = {\n        metadata: element.metadata,\n        extensions: element.extensions,\n        stats: {\n          totalEntries: stats.totalEntries,\n          totalSize: stats.totalSize,\n          oldestEntry: stats.oldestEntry?.toISOString(),\n          newestEntry: stats.newestEntry?.toISOString(),\n          topTags: Array.from(stats.tagFrequency.entries())\n            .sort((a, b) => b[1] - a[1])\n            .slice(0, 10)\n            .map(([tag, count]) => ({ tag, count }))\n        },\n        entries: JSON.parse(element.serialize()).entries\n      };\n      \n      // SECURITY FIX: Use secure YAML dumping with safety options\n      // Previously: Could allow dangerous YAML features\n      // Now: Uses FAILSAFE_SCHEMA and security options to prevent code execution\n      const yamlContent = yaml.dump(data, {\n        schema: yaml.FAILSAFE_SCHEMA,\n        noRefs: true,\n        skipInvalid: true,\n        sortKeys: true\n      });\n      \n      // CRITICAL FIX: Use atomic file write to prevent corruption\n      // Previously: await fs.writeFile(fullPath, yamlContent, 'utf-8');\n      // Now: Uses FileLockManager for atomic write with proper encoding\n      await FileLockManager.atomicWriteFile(fullPath, yamlContent, { encoding: 'utf-8' });\n      \n      // Update cache\n      this.memoryCache.set(fullPath, element);\n\n      // Update content hash index\n      this.contentHashIndex.set(contentHash, fullPath);\n\n      // Log successful save\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_SAVED,\n        severity: 'LOW',\n        source: 'MemoryManager.save',\n        details: `Saved memory to ${path.basename(fullPath)} with ${stats.totalEntries} entries`\n      });\n      \n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_SAVE_FAILED,\n        severity: 'HIGH',\n        source: 'MemoryManager.save',\n        details: `Failed to save memory to ${filePath}: ${error}`\n      });\n      throw new Error(`Failed to save memory: ${error}`);\n    }\n  }\n  \n  /**\n   * List all available memories\n   */\n  async list(): Promise<Memory[]> {\n    const memories: Memory[] = [];\n\n    try {\n      // Get all date folders\n      const dateFolders = await this.getDateFolders();\n\n      // Also check root directory for any memory files\n      // Memory files should be .yaml format only\n      const rootFiles = await fs.readdir(this.memoriesDir)\n        .then(files => files.filter(f => f.endsWith('.yaml')))\n        .catch(() => []);\n\n      // Process root files first (legacy)\n      for (const file of rootFiles) {\n        try {\n          const memory = await this.load(file);\n          memories.push(memory);\n        } catch (error) {\n          SecurityMonitor.logSecurityEvent({\n            type: MEMORY_SECURITY_EVENTS.MEMORY_LIST_ITEM_FAILED,\n            severity: 'LOW',\n            source: 'MemoryManager.list',\n            details: `Failed to load ${file}: ${error}`\n          });\n        }\n      }\n\n      // Process date folders\n      for (const dateFolder of dateFolders) {\n        const folderPath = path.join(this.memoriesDir, dateFolder);\n        const files = await fs.readdir(folderPath)\n          .then(files => files.filter(f => f.endsWith('.yaml')))\n          .catch(() => []);\n\n        for (const file of files) {\n          try {\n            const memory = await this.load(path.join(dateFolder, file));\n            memories.push(memory);\n          } catch (error) {\n            SecurityMonitor.logSecurityEvent({\n              type: MEMORY_SECURITY_EVENTS.MEMORY_LIST_ITEM_FAILED,\n              severity: 'LOW',\n              source: 'MemoryManager.list',\n              details: `Failed to load ${dateFolder}/${file}: ${error}`\n            });\n          }\n        }\n      }\n\n      return memories;\n      \n    } catch (error) {\n      if ((error as any).code === 'ENOENT') {\n        // Directory doesn't exist yet\n        return [];\n      }\n      throw error;\n    }\n  }\n  \n  /**\n   * Find memories matching a predicate\n   */\n  async find(predicate: (element: Memory) => boolean): Promise<Memory | undefined> {\n    const memories = await this.list();\n    return memories.find(predicate);\n  }\n  \n  /**\n   * Find multiple memories matching a predicate\n   */\n  async findMany(predicate: (element: Memory) => boolean): Promise<Memory[]> {\n    const memories = await this.list();\n    return memories.filter(predicate);\n  }\n  \n  /**\n   * Delete a memory file\n   * SECURITY: Validates path and logs deletion\n   */\n  async delete(filePath: string): Promise<void> {\n    try {\n      const fullPath = await this.validateAndResolvePath(filePath);\n      \n      // Check if file exists\n      await fs.access(fullPath);\n      \n      // Delete the file\n      await fs.unlink(fullPath);\n      \n      // Remove from cache\n      this.memoryCache.delete(fullPath);\n      \n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_DELETED,\n        severity: 'MEDIUM',\n        source: 'MemoryManager.delete',\n        details: `Deleted memory file: ${path.basename(fullPath)}`\n      });\n      \n    } catch (error) {\n      if ((error as any).code === 'ENOENT') {\n        // File doesn't exist, not an error for delete operation\n        return;\n      }\n      throw error;\n    }\n  }\n  \n  /**\n   * Check if a memory file exists\n   */\n  async exists(filePath: string): Promise<boolean> {\n    try {\n      const fullPath = await this.validateAndResolvePath(filePath);\n      await fs.access(fullPath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Create a new memory with metadata\n   */\n  async create(metadata: Partial<MemoryMetadata>): Promise<Memory> {\n    return new Memory(metadata);\n  }\n  \n  /**\n   * Import a memory from JSON/YAML string\n   * SECURITY: Full validation of imported content\n   * @param data JSON or YAML string containing memory data\n   * @param format Format of the input data ('json' or 'yaml')\n   * @returns Promise resolving to the imported Memory instance\n   * @throws {Error} When JSON/YAML parsing fails\n   * @throws {Error} When imported data is missing required fields\n   * @throws {Error} When YAML content exceeds maximum allowed size\n   * @throws {Error} When imported memory fails validation\n   */\n  async importElement(data: string, format: 'json' | 'yaml' = 'yaml'): Promise<Memory> {\n    try {\n      let parsed: any;\n      \n      if (format === 'json') {\n        parsed = JSON.parse(data);\n      } else {\n        // HIGH SEVERITY FIX: Use secure YAML parsing\n        // Memory import expects pure YAML (not frontmatter), so we parse it securely\n        try {\n          // First validate the YAML content size\n          if (data.length > MEMORY_CONSTANTS.MAX_YAML_SIZE) {\n            throw new Error('YAML content exceeds maximum allowed size');\n          }\n          \n          // Create a wrapper to use SecureYamlParser with pure YAML\n          // Add minimal frontmatter markers to satisfy parser\n          const wrappedYaml = `---\\n${data}\\n---\\n`;\n          \n          const parseResult = SecureYamlParser.parse(wrappedYaml, {\n            maxYamlSize: MEMORY_CONSTANTS.MAX_YAML_SIZE,\n            validateContent: true\n          });\n          \n          // Extract the parsed data (will be in the 'data' property)\n          parsed = parseResult.data;\n          \n        } catch (yamlError) {\n          throw new Error(`Invalid YAML: ${yamlError}`);\n        }\n        \n        // Validate it's an object\n        if (!parsed || typeof parsed !== 'object') {\n          throw new Error('YAML must contain an object');\n        }\n      }\n      \n      // Handle different structures from YAML parsing\n      let metadata = parsed.metadata;\n      let entries = parsed.entries || (parsed.data && parsed.data.entries);\n      \n      // Validate required fields\n      if (!metadata || !metadata.name) {\n        throw new Error('Memory must have metadata with name');\n      }\n      \n      // Create memory instance\n      const memory = new Memory(metadata);\n      \n      // Load entries if present\n      if (entries) {\n        memory.deserialize(JSON.stringify({\n          id: memory.id,\n          type: memory.type,\n          version: memory.version,\n          metadata: memory.metadata,\n          extensions: memory.extensions,\n          entries: entries\n        }));\n      }\n      \n      return memory;\n      \n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_IMPORT_FAILED,\n        severity: 'MEDIUM',\n        source: 'MemoryManager.importElement',\n        details: `Failed to import memory: ${error}`\n      });\n      throw new Error(`Failed to import memory: ${error}`);\n    }\n  }\n  \n  /**\n   * Export a memory to YAML string\n   */\n  async exportElement(element: Memory): Promise<string> {\n    const stats = element.getStats();\n    const data = {\n      metadata: element.metadata,\n      extensions: element.extensions,\n      stats: {\n        totalEntries: stats.totalEntries,\n        totalSize: stats.totalSize,\n        oldestEntry: stats.oldestEntry?.toISOString(),\n        newestEntry: stats.newestEntry?.toISOString()\n      },\n      entries: JSON.parse(element.serialize()).entries\n    };\n    \n    // SECURITY FIX: Use secure YAML dumping\n    return yaml.dump(data, {\n      schema: yaml.FAILSAFE_SCHEMA,\n      noRefs: true,\n      skipInvalid: true,\n      sortKeys: true\n    });\n  }\n  \n  /**\n   * Validate a memory element\n   */\n  validate(element: Memory): ElementValidationResult {\n    return element.validate();\n  }\n  \n  /**\n   * Validate and resolve a file path\n   * SECURITY: Prevents directory traversal attacks\n   */\n  validatePath(filePath: string): boolean {\n    try {\n      // Perform synchronous validation checks\n      const normalized = path.normalize(filePath);\n      \n      // Check for path traversal attempts\n      if (normalized.includes('..') || path.isAbsolute(normalized)) {\n        return false;\n      }\n      \n      // Ensure proper extension\n      if (!normalized.endsWith('.md') && !normalized.endsWith('.yaml') && !normalized.endsWith('.yml')) {\n        return false;\n      }\n      \n      return true;\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Get the element type this manager handles\n   */\n  getElementType(): ElementType {\n    return ElementType.MEMORY;\n  }\n  \n  /**\n   * Get the file extension for memory files\n   */\n  getFileExtension(): string {\n    return '.yaml';\n  }\n  \n  // Private helper methods\n  \n  /**\n   * Validate and resolve a file path to prevent security issues\n   * @param filePath Path to validate and resolve\n   * @returns Promise resolving to the validated full path\n   * @throws {Error} When path contains traversal attempts (../)\n   * @throws {Error} When path is absolute or invalid\n   * @throws {Error} When file extension is not allowed (.md, .yaml, .yml)\n   * @throws {Error} When resolved path would be outside memories directory\n   */\n  private async validateAndResolvePath(filePath: string): Promise<string> {\n    // SECURITY FIX: Comprehensive path validation\n    // Enhanced validation inspired by Jeet Singh (@jeetsingh008) - PR #1035\n\n    // First normalize the path to resolve any ./ or ../ sequences\n    const normalized = path.normalize(filePath);\n\n    // Check for path traversal attempts - both in original and normalized\n    // Check both the normalized path and the original for any traversal patterns\n    if (normalized.includes('..') ||\n        filePath.includes('..') ||\n        path.isAbsolute(normalized) ||\n        path.isAbsolute(filePath)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'HIGH',\n        source: 'MemoryManager.validateAndResolvePath',\n        details: `Blocked path traversal attempt: ${filePath}`\n      });\n      throw new Error('Invalid file path: Path traversal detected');\n    }\n    \n    // Ensure proper extension - memories should only be .yaml or .yml\n    if (!normalized.endsWith('.yaml') && !normalized.endsWith('.yml')) {\n      throw new Error('Memory files must have .yaml or .yml extension');\n    }\n    \n    // Construct full path\n    const fullPath = path.join(this.memoriesDir, normalized);\n    \n    // Verify it's within memories directory\n    const relative = path.relative(this.memoriesDir, fullPath);\n    if (relative.startsWith('..') || path.isAbsolute(relative)) {\n      throw new Error('File path must be within memories directory');\n    }\n    \n    return fullPath;\n  }\n  \n  private parseMemoryFile(parsed: any): { metadata: MemoryMetadata; content: string } {\n    // FIX: SecureYamlParser returns data in 'data' property, not 'metadata'\n    // For markdown files with YAML frontmatter, the structure is:\n    // parsed.data = YAML frontmatter values\n    // parsed.content = markdown content after frontmatter\n\n    // For pure YAML memory files, we need to check if metadata is directly in data\n    const yamlData = parsed.data || {};\n\n    // Memory files saved by the system have metadata as a top-level key\n    const metadataSource = yamlData.metadata || yamlData;\n\n    // Extract metadata with validation\n    const metadata: MemoryMetadata = {\n      name: sanitizeInput(metadataSource.name || 'Unnamed Memory', 100),\n      description: metadataSource.description ?\n        sanitizeInput(metadataSource.description, 500) :\n        '',\n      version: metadataSource.version || '1.0.0',\n      author: metadataSource.author,\n      created: metadataSource.created,\n      modified: metadataSource.modified || new Date().toISOString(),\n      tags: Array.isArray(metadataSource.tags) ?\n        metadataSource.tags.map((tag: string) => sanitizeInput(tag, MEMORY_CONSTANTS.MAX_TAG_LENGTH)) :\n        [],\n      storageBackend: metadataSource.storage_backend || metadataSource.storageBackend || MEMORY_CONSTANTS.DEFAULT_STORAGE_BACKEND,\n      retentionDays: metadataSource.retention_policy?.default ?\n        this.parseRetentionDays(metadataSource.retention_policy.default) :\n        (metadataSource.retentionDays || MEMORY_CONSTANTS.DEFAULT_RETENTION_DAYS),\n      privacyLevel: metadataSource.privacy_level || metadataSource.privacyLevel || MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL,\n      searchable: metadataSource.searchable !== false,\n      maxEntries: metadataSource.maxEntries || MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT\n    };\n\n    // Extract content (if any)\n    const content = parsed.content || '';\n\n    return { metadata, content };\n  }\n\n  /**\n   * Helper to parse retention days from various formats\n   */\n  private parseRetentionDays(retention: string | number): number {\n    if (typeof retention === 'number') return retention;\n    if (retention === 'permanent' || retention === 'perpetual') return 999999;\n    const match = retention.match(/(\\d+)\\s*days?/i);\n    return match ? parseInt(match[1]) : MEMORY_CONSTANTS.DEFAULT_RETENTION_DAYS;\n  }\n}"]}
@@ -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 = "1.9.6";
6
- export declare const BUILD_TIMESTAMP = "2025-09-20T16:07:40.778Z";
5
+ export declare const PACKAGE_VERSION = "1.9.7";
6
+ export declare const BUILD_TIMESTAMP = "2025-09-20T17:09:49.524Z";
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 = '1.9.6';
6
- export const BUILD_TIMESTAMP = '2025-09-20T16:07:40.778Z';
5
+ export const PACKAGE_VERSION = '1.9.7';
6
+ export const BUILD_TIMESTAMP = '2025-09-20T17:09:49.524Z';
7
7
  export const BUILD_TYPE = 'npm';
8
8
  export const PACKAGE_NAME = '@dollhousemcp/mcp-server';
9
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcxLjkuNic7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjUtMDktMjBUMTY6MDc6NDAuNzc4Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcxLjkuNyc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjUtMDktMjBUMTc6MDk6NDkuNTI0Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
@@ -1 +1 @@
1
- {"version":3,"file":"PortfolioManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioManager.ts"],"names":[],"mappings":"AAAA;;GAEG;AAMH,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAS1D,OAAO,EAAE,WAAW,EAAE,CAAC;AACvB,YAAY,EAAE,eAAe,EAAE,CAAC;AAEhC,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAmB;IAC1C,OAAO,CAAC,MAAM,CAAC,YAAY,CAAS;IACpC,OAAO,CAAC,MAAM,CAAC,kBAAkB,CAAS;IAC1C,OAAO,CAAC,MAAM,CAAC,qBAAqB,CAA8B;IAClE,OAAO,CAAC,OAAO,CAAS;IAExB,OAAO;WA4BO,WAAW,CAAC,MAAM,CAAC,EAAE,eAAe,GAAG,gBAAgB;IAiBrE;;OAEG;IACI,UAAU,IAAI,MAAM;IAI3B;;OAEG;IACI,aAAa,CAAC,IAAI,EAAE,WAAW,GAAG,MAAM;IAI/C;;;OAGG;IACU,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAuBxC;;OAEG;YACW,qBAAqB;IAiDnC;;OAEG;IACU,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC;IASvC;;;OAGG;IACI,aAAa,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAgD/C;;OAEG;IACU,YAAY,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAoC/D;;OAEG;IACI,cAAc,CAAC,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,GAAG,MAAM;IA0ClE;;OAEG;IACU,aAAa,CAAC,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IASjF;;OAEG;IACI,oBAAoB,IAAI,MAAM;IAIrC;;OAEG;IACU,iBAAiB,IAAI,OAAO,CAAC,OAAO,CAAC;IAUlD;;OAEG;IACU,aAAa,IAAI,OAAO,CAAC,MAAM,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;IAWlE;;;OAGG;YACW,8BAA8B;CA+D7C"}
1
+ {"version":3,"file":"PortfolioManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioManager.ts"],"names":[],"mappings":"AAAA;;GAEG;AAMH,OAAO,EAAE,WAAW,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAmB1D,OAAO,EAAE,WAAW,EAAE,CAAC;AACvB,YAAY,EAAE,eAAe,EAAE,CAAC;AAEhC,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAmB;IAC1C,OAAO,CAAC,MAAM,CAAC,YAAY,CAAS;IACpC,OAAO,CAAC,MAAM,CAAC,kBAAkB,CAAS;IAC1C,OAAO,CAAC,MAAM,CAAC,qBAAqB,CAA8B;IAClE,OAAO,CAAC,OAAO,CAAS;IAExB,OAAO;WA4BO,WAAW,CAAC,MAAM,CAAC,EAAE,eAAe,GAAG,gBAAgB;IAiBrE;;OAEG;IACI,UAAU,IAAI,MAAM;IAI3B;;OAEG;IACI,aAAa,CAAC,IAAI,EAAE,WAAW,GAAG,MAAM;IAI/C;;;OAGG;IACU,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAuBxC;;OAEG;YACW,qBAAqB;IAiDnC;;OAEG;IACU,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC;IASvC;;;OAGG;IACI,aAAa,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAgD/C;;OAEG;IACU,YAAY,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAqC/D;;OAEG;IACI,cAAc,CAAC,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,GAAG,MAAM;IA0ClE;;OAEG;IACU,aAAa,CAAC,IAAI,EAAE,WAAW,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IASjF;;OAEG;IACI,oBAAoB,IAAI,MAAM;IAIrC;;OAEG;IACU,iBAAiB,IAAI,OAAO,CAAC,OAAO,CAAC;IAUlD;;OAEG;IACU,aAAa,IAAI,OAAO,CAAC,MAAM,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;IAWlE;;;OAGG;YACW,8BAA8B;CA+D7C"}
@@ -11,7 +11,16 @@ import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
11
11
  import { DefaultElementProvider } from './DefaultElementProvider.js';
12
12
  import { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';
13
13
  // Constants
14
- const ELEMENT_FILE_EXTENSION = '.md';
14
+ const ELEMENT_FILE_EXTENSIONS = {
15
+ [ElementType.PERSONA]: '.md',
16
+ [ElementType.SKILL]: '.md',
17
+ [ElementType.TEMPLATE]: '.md',
18
+ [ElementType.AGENT]: '.md',
19
+ [ElementType.MEMORY]: '.yaml',
20
+ [ElementType.ENSEMBLE]: '.md'
21
+ };
22
+ // Default extension for backward compatibility
23
+ const DEFAULT_ELEMENT_FILE_EXTENSION = '.md';
15
24
  export { ElementType };
16
25
  export class PortfolioManager {
17
26
  static instance;
@@ -206,11 +215,12 @@ export class PortfolioManager {
206
215
  */
207
216
  async listElements(type) {
208
217
  const elementDir = this.getElementDir(type);
218
+ const fileExtension = ELEMENT_FILE_EXTENSIONS[type] || DEFAULT_ELEMENT_FILE_EXTENSION;
209
219
  try {
210
220
  const files = await fs.readdir(elementDir);
211
- // Filter for markdown files only and exclude test elements
221
+ // Filter for correct file extension based on element type and exclude test elements
212
222
  return files
213
- .filter(file => file.endsWith(ELEMENT_FILE_EXTENSION))
223
+ .filter(file => file.endsWith(fileExtension))
214
224
  .filter(file => !this.isTestElement(file));
215
225
  }
216
226
  catch (error) {
@@ -377,4 +387,4 @@ export class PortfolioManager {
377
387
  }
378
388
  }
379
389
  }
380
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PortfolioManager.js","sourceRoot":"","sources":["../../src/portfolio/PortfolioManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Portfolio Manager - Manages the portfolio directory structure for all element types\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { homedir } from 'os';\nimport { logger } from '../utils/logger.js';\nimport { ElementType, PortfolioConfig } from './types.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { DefaultElementProvider } from './DefaultElementProvider.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\n\n// Constants\nconst ELEMENT_FILE_EXTENSION = '.md';\n\nexport { ElementType };\nexport type { PortfolioConfig };\n\nexport class PortfolioManager {\n  private static instance: PortfolioManager;\n  private static instanceLock = false;\n  private static initializationLock = false;\n  private static initializationPromise: Promise<void> | null = null;\n  private baseDir: string;\n  \n  private constructor(config?: PortfolioConfig) {\n    // Get potential directory from environment or config\n    const envDir = process.env.DOLLHOUSE_PORTFOLIO_DIR;\n    const configDir = config?.baseDir;\n    const defaultDir = path.join(homedir(), '.dollhouse', 'portfolio');\n    \n    // Validate environment variable if provided\n    if (envDir) {\n      if (!path.isAbsolute(envDir)) {\n        throw new Error('DOLLHOUSE_PORTFOLIO_DIR must be an absolute path');\n      }\n      // Additional validation for suspicious paths\n      if (envDir.includes('..') || envDir.startsWith('/etc') || envDir.startsWith('/sys')) {\n        throw new Error('DOLLHOUSE_PORTFOLIO_DIR contains suspicious path segments');\n      }\n    }\n    \n    // Validate config directory if provided\n    if (configDir && !path.isAbsolute(configDir)) {\n      throw new Error('Portfolio config baseDir must be an absolute path');\n    }\n    \n    // Use environment variable if set, otherwise config, otherwise default\n    this.baseDir = envDir || configDir || defaultDir;\n    \n    logger.info(`[PortfolioManager] Portfolio base directory: ${this.baseDir}`);\n  }\n  \n  public static getInstance(config?: PortfolioConfig): PortfolioManager {\n    if (!PortfolioManager.instance) {\n      // Check if another thread is already creating the instance\n      if (PortfolioManager.instanceLock) {\n        throw new Error('PortfolioManager instance is being created by another thread');\n      }\n      \n      try {\n        PortfolioManager.instanceLock = true;\n        PortfolioManager.instance = new PortfolioManager(config);\n      } finally {\n        PortfolioManager.instanceLock = false;\n      }\n    }\n    return PortfolioManager.instance;\n  }\n  \n  /**\n   * Get the base portfolio directory\n   */\n  public getBaseDir(): string {\n    return this.baseDir;\n  }\n  \n  /**\n   * Get the directory for a specific element type\n   */\n  public getElementDir(type: ElementType): string {\n    return path.join(this.baseDir, type);\n  }\n  \n  /**\n   * Initialize the portfolio directory structure\n   * Uses locking to prevent race conditions during concurrent initialization\n   */\n  public async initialize(): Promise<void> {\n    // If already initializing, wait for the existing initialization\n    if (PortfolioManager.initializationPromise) {\n      return PortfolioManager.initializationPromise;\n    }\n    \n    // If already initialized, check if directories exist\n    if (await this.exists()) {\n      logger.debug('[PortfolioManager] Portfolio already initialized');\n      return;\n    }\n    \n    // Create initialization promise to prevent concurrent initialization\n    PortfolioManager.initializationPromise = this.performInitialization();\n    \n    try {\n      await PortfolioManager.initializationPromise;\n    } finally {\n      // Clear the promise after completion\n      PortfolioManager.initializationPromise = null;\n    }\n  }\n  \n  /**\n   * Perform the actual initialization - should only be called once\n   */\n  private async performInitialization(): Promise<void> {\n    logger.info('[PortfolioManager] Initializing portfolio directory structure');\n    \n    // Create base directory\n    try {\n      await fs.mkdir(this.baseDir, { recursive: true });\n    } catch (error) {\n      const err = error as NodeJS.ErrnoException;\n      // In read-only environments (like Docker), we can't create directories\n      // Log but continue - the portfolio will be empty but functional\n      if (err.code === 'EACCES' || err.code === 'EROFS' || err.code === 'ENOENT') {\n        logger.warn(`[PortfolioManager] Cannot create portfolio directory (read-only environment?): ${err.message}`);\n        logger.info(`[DollhouseMCP] Running in read-only mode - portfolio features disabled`);\n        return;\n      }\n      throw error;\n    }\n    \n    // Create subdirectories for each element type\n    for (const elementType of Object.values(ElementType)) {\n      const elementDir = path.join(this.baseDir, elementType);\n      await fs.mkdir(elementDir, { recursive: true });\n      logger.debug(`[PortfolioManager] Created directory: ${elementDir}`);\n    }\n    \n    // Create special directories for stateful elements\n    const agentStateDir = path.join(this.baseDir, ElementType.AGENT, '.state');\n    await fs.mkdir(agentStateDir, { recursive: true });\n    \n    logger.info('[PortfolioManager] Portfolio directory structure initialized');\n    \n    // Migration for v1.4.2 users: rename singular directories to plural\n    await this.migrateFromSingularDirectories();\n    \n    // Populate with default elements if this is a new installation\n    // Skip during tests to avoid interference\n    if (process.env.NODE_ENV !== 'test') {\n      try {\n        const defaultProvider = new DefaultElementProvider();\n        await defaultProvider.populateDefaults(this.baseDir);\n      } catch (error) {\n        logger.error('[PortfolioManager] Error populating default elements:', error);\n        // Log to stderr for Claude Desktop visibility\n        logger.error(`[PortfolioManager] CRITICAL: Failed to populate default elements: ${error instanceof Error ? error.message : String(error)}`);\n        // Continue anyway - empty portfolio is valid\n      }\n    }\n  }\n  \n  /**\n   * Check if portfolio directory exists\n   */\n  public async exists(): Promise<boolean> {\n    try {\n      await fs.access(this.baseDir);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Check if a filename appears to be a test element\n   * SAFETY: Pattern-based filtering only, no content parsing\n   */\n  public isTestElement(filename: string): boolean {\n    // Dangerous test patterns that should never appear in production\n    const dangerousPatterns = [\n      /^bin-sh/i,\n      /^rm-rf/i,\n      /^nc-e-bin/i,\n      /^python-c-import/i,\n      /^curl.*evil/i,\n      /^wget.*malicious/i,\n      /^eval-/i,\n      /^exec-/i,\n      /^bash-c-/i,\n      /^sh-c-/i,\n      /^powershell-/i,\n      /^cmd-c-/i,\n      /shell-injection/i\n    ];\n    \n    // Common test patterns\n    const testPatterns = [\n      /^test-/i,\n      /^memory-test-/i,\n      /^yaml-test/i,\n      /^perf-test-/i,\n      /^stability-test-/i,\n      /^roundtrip-test/i,\n      /test-persona/i,\n      /test-skill/i,\n      /test-template/i,\n      /test-agent/i,\n      /\\.test\\./,\n      /__test__/,\n      /test-data/,\n      /penetration-test/i,\n      /metadata-test/i,\n      /testpersona\\d+/i  // Generated test personas with timestamps\n    ];\n    \n    // Check dangerous patterns first\n    if (dangerousPatterns.some(pattern => pattern.test(filename))) {\n      logger.warn(`[PortfolioManager] Filtered dangerous test element: ${filename}`);\n      return true;\n    }\n    \n    // Check common test patterns\n    return testPatterns.some(pattern => pattern.test(filename));\n  }\n\n  /**\n   * List all elements of a specific type\n   */\n  public async listElements(type: ElementType): Promise<string[]> {\n    const elementDir = this.getElementDir(type);\n    \n    try {\n      const files = await fs.readdir(elementDir);\n      // Filter for markdown files only and exclude test elements\n      return files\n        .filter(file => file.endsWith(ELEMENT_FILE_EXTENSION))\n        .filter(file => !this.isTestElement(file));\n    } catch (error) {\n      const err = error as NodeJS.ErrnoException;\n      \n      if (err.code === 'ENOENT') {\n        // Directory doesn't exist yet - this is expected for new installations\n        logger.debug(`[PortfolioManager] Element directory doesn't exist yet: ${elementDir}`);\n        return [];\n      }\n      \n      if (err.code === 'EACCES' || err.code === 'EPERM') {\n        // Permission denied - log but return empty array\n        ErrorHandler.logError('PortfolioManager.listElements', error, { elementDir });\n        return [];\n      }\n      \n      if (err.code === 'ENOTDIR') {\n        // Path exists but is not a directory\n        ErrorHandler.logError('PortfolioManager.listElements', error, { elementDir });\n        throw ErrorHandler.createError(`Path is not a directory: ${elementDir}`, ErrorCategory.SYSTEM_ERROR);\n      }\n      \n      // For any other errors, throw with context\n      ErrorHandler.logError('PortfolioManager.listElements', error, { elementDir });\n      throw ErrorHandler.wrapError(error, 'Failed to list elements', ErrorCategory.SYSTEM_ERROR);\n    }\n  }\n  \n  /**\n   * Get full path to an element file\n   */\n  public getElementPath(type: ElementType, filename: string): string {\n    // Validate filename to prevent path traversal\n    if (!filename || typeof filename !== 'string') {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'MEDIUM',\n        source: 'PortfolioManager.getElementPath',\n        details: `Invalid filename provided: ${typeof filename}`,\n        additionalData: { elementType: type, filename: String(filename) }\n      });\n      throw new Error('Invalid filename: must be a non-empty string');\n    }\n    \n    // Check for path traversal attempts\n    if (filename.includes('..') || filename.includes('/') || filename.includes('\\\\') || path.isAbsolute(filename)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'HIGH',\n        source: 'PortfolioManager.getElementPath',\n        details: `Path traversal attempt detected in filename: ${filename}`,\n        additionalData: { elementType: type, filename }\n      });\n      throw new Error(`Invalid filename: contains path traversal characters: ${filename}`);\n    }\n    \n    // Additional validation for hidden files and special characters\n    if (filename.startsWith('.') || filename.includes('\\0')) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'MEDIUM',\n        source: 'PortfolioManager.getElementPath',\n        details: `Invalid filename characters detected: ${filename}`,\n        additionalData: { elementType: type, filename, hasHiddenFile: filename.startsWith('.'), hasNullByte: filename.includes('\\0') }\n      });\n      throw new Error(`Invalid filename: contains invalid characters: ${filename}`);\n    }\n    \n    // Ensure filename ends with .md\n    const safeFilename = filename.endsWith('.md') ? filename : `${filename}.md`;\n    return path.join(this.getElementDir(type), safeFilename);\n  }\n  \n  /**\n   * Check if an element exists\n   */\n  public async elementExists(type: ElementType, filename: string): Promise<boolean> {\n    try {\n      await fs.access(this.getElementPath(type, filename));\n      return true;\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Get legacy personas directory path (for migration)\n   */\n  public getLegacyPersonasDir(): string {\n    return path.join(homedir(), '.dollhouse', 'personas');\n  }\n  \n  /**\n   * Check if legacy personas directory exists\n   */\n  public async hasLegacyPersonas(): Promise<boolean> {\n    try {\n      await fs.access(this.getLegacyPersonasDir());\n      const files = await fs.readdir(this.getLegacyPersonasDir());\n      return files.some(file => file.endsWith('.md'));\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Get portfolio statistics\n   */\n  public async getStatistics(): Promise<Record<ElementType, number>> {\n    const stats: Record<string, number> = {};\n    \n    for (const elementType of Object.values(ElementType)) {\n      const elements = await this.listElements(elementType);\n      stats[elementType] = elements.length;\n    }\n    \n    return stats as Record<ElementType, number>;\n  }\n  \n  /**\n   * Migrate from v1.4.2 singular directory names to v1.4.3 plural names\n   * This handles the upgrade path for existing users\n   */\n  private async migrateFromSingularDirectories(): Promise<void> {\n    const oldToNew: Record<string, string> = {\n      'persona': 'personas',\n      'skill': 'skills',\n      'template': 'templates',\n      'agent': 'agents',\n      'memory': 'memories',\n      'ensemble': 'ensembles'\n    };\n    \n    for (const [oldName, newName] of Object.entries(oldToNew)) {\n      // Unicode normalize the directory names (even though they're hardcoded, for security audit)\n      const normalizedOld = UnicodeValidator.normalize(oldName);\n      const normalizedNew = UnicodeValidator.normalize(newName);\n      \n      if (!normalizedOld.isValid || !normalizedNew.isValid) {\n        // This should never happen with our hardcoded values, but for completeness\n        logger.error(`[PortfolioManager] Invalid Unicode in directory names during migration`);\n        continue;\n      }\n      \n      const oldDir = path.join(this.baseDir, normalizedOld.normalizedContent);\n      const newDir = path.join(this.baseDir, normalizedNew.normalizedContent);\n      \n      try {\n        // Check if old directory exists\n        await fs.access(oldDir);\n        \n        // Check if new directory already has content\n        try {\n          const newDirFiles = await fs.readdir(newDir);\n          if (newDirFiles.length > 0) {\n            logger.warn(\n              `[PortfolioManager] Both ${oldName} and ${newName} directories exist. Keeping ${newName}, skipping migration.`,\n              { oldDir, newDir, fileCount: newDirFiles.length }\n            );\n            continue;\n          }\n        } catch {\n          // New directory doesn't exist or is empty, proceed with migration\n        }\n        \n        // Perform the migration\n        logger.info(`[PortfolioManager] Migrating ${oldName} → ${newName}`);\n        await fs.rename(oldDir, newDir);\n        \n        // Log security event for audit trail\n        SecurityMonitor.logSecurityEvent({\n          type: 'DIRECTORY_MIGRATION',\n          severity: 'LOW',\n          source: 'PortfolioManager.migrateFromSingularDirectories',\n          details: `Migrated directory from ${oldName} to ${newName} for v1.4.3 compatibility`,\n          metadata: { oldDir, newDir }\n        });\n        \n      } catch (error) {\n        // Old directory doesn't exist, which is fine\n        if ((error as any).code !== 'ENOENT') {\n          logger.error(`[PortfolioManager] Error during migration of ${oldName}:`, error);\n        }\n      }\n    }\n  }\n}"]}
390
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PortfolioManager.js","sourceRoot":"","sources":["../../src/portfolio/PortfolioManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Portfolio Manager - Manages the portfolio directory structure for all element types\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { homedir } from 'os';\nimport { logger } from '../utils/logger.js';\nimport { ElementType, PortfolioConfig } from './types.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { DefaultElementProvider } from './DefaultElementProvider.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\n\n// Constants\nconst ELEMENT_FILE_EXTENSIONS: Record<ElementType, string> = {\n  [ElementType.PERSONA]: '.md',\n  [ElementType.SKILL]: '.md',\n  [ElementType.TEMPLATE]: '.md',\n  [ElementType.AGENT]: '.md',\n  [ElementType.MEMORY]: '.yaml',\n  [ElementType.ENSEMBLE]: '.md'\n};\n\n// Default extension for backward compatibility\nconst DEFAULT_ELEMENT_FILE_EXTENSION = '.md';\n\nexport { ElementType };\nexport type { PortfolioConfig };\n\nexport class PortfolioManager {\n  private static instance: PortfolioManager;\n  private static instanceLock = false;\n  private static initializationLock = false;\n  private static initializationPromise: Promise<void> | null = null;\n  private baseDir: string;\n  \n  private constructor(config?: PortfolioConfig) {\n    // Get potential directory from environment or config\n    const envDir = process.env.DOLLHOUSE_PORTFOLIO_DIR;\n    const configDir = config?.baseDir;\n    const defaultDir = path.join(homedir(), '.dollhouse', 'portfolio');\n    \n    // Validate environment variable if provided\n    if (envDir) {\n      if (!path.isAbsolute(envDir)) {\n        throw new Error('DOLLHOUSE_PORTFOLIO_DIR must be an absolute path');\n      }\n      // Additional validation for suspicious paths\n      if (envDir.includes('..') || envDir.startsWith('/etc') || envDir.startsWith('/sys')) {\n        throw new Error('DOLLHOUSE_PORTFOLIO_DIR contains suspicious path segments');\n      }\n    }\n    \n    // Validate config directory if provided\n    if (configDir && !path.isAbsolute(configDir)) {\n      throw new Error('Portfolio config baseDir must be an absolute path');\n    }\n    \n    // Use environment variable if set, otherwise config, otherwise default\n    this.baseDir = envDir || configDir || defaultDir;\n    \n    logger.info(`[PortfolioManager] Portfolio base directory: ${this.baseDir}`);\n  }\n  \n  public static getInstance(config?: PortfolioConfig): PortfolioManager {\n    if (!PortfolioManager.instance) {\n      // Check if another thread is already creating the instance\n      if (PortfolioManager.instanceLock) {\n        throw new Error('PortfolioManager instance is being created by another thread');\n      }\n      \n      try {\n        PortfolioManager.instanceLock = true;\n        PortfolioManager.instance = new PortfolioManager(config);\n      } finally {\n        PortfolioManager.instanceLock = false;\n      }\n    }\n    return PortfolioManager.instance;\n  }\n  \n  /**\n   * Get the base portfolio directory\n   */\n  public getBaseDir(): string {\n    return this.baseDir;\n  }\n  \n  /**\n   * Get the directory for a specific element type\n   */\n  public getElementDir(type: ElementType): string {\n    return path.join(this.baseDir, type);\n  }\n  \n  /**\n   * Initialize the portfolio directory structure\n   * Uses locking to prevent race conditions during concurrent initialization\n   */\n  public async initialize(): Promise<void> {\n    // If already initializing, wait for the existing initialization\n    if (PortfolioManager.initializationPromise) {\n      return PortfolioManager.initializationPromise;\n    }\n    \n    // If already initialized, check if directories exist\n    if (await this.exists()) {\n      logger.debug('[PortfolioManager] Portfolio already initialized');\n      return;\n    }\n    \n    // Create initialization promise to prevent concurrent initialization\n    PortfolioManager.initializationPromise = this.performInitialization();\n    \n    try {\n      await PortfolioManager.initializationPromise;\n    } finally {\n      // Clear the promise after completion\n      PortfolioManager.initializationPromise = null;\n    }\n  }\n  \n  /**\n   * Perform the actual initialization - should only be called once\n   */\n  private async performInitialization(): Promise<void> {\n    logger.info('[PortfolioManager] Initializing portfolio directory structure');\n    \n    // Create base directory\n    try {\n      await fs.mkdir(this.baseDir, { recursive: true });\n    } catch (error) {\n      const err = error as NodeJS.ErrnoException;\n      // In read-only environments (like Docker), we can't create directories\n      // Log but continue - the portfolio will be empty but functional\n      if (err.code === 'EACCES' || err.code === 'EROFS' || err.code === 'ENOENT') {\n        logger.warn(`[PortfolioManager] Cannot create portfolio directory (read-only environment?): ${err.message}`);\n        logger.info(`[DollhouseMCP] Running in read-only mode - portfolio features disabled`);\n        return;\n      }\n      throw error;\n    }\n    \n    // Create subdirectories for each element type\n    for (const elementType of Object.values(ElementType)) {\n      const elementDir = path.join(this.baseDir, elementType);\n      await fs.mkdir(elementDir, { recursive: true });\n      logger.debug(`[PortfolioManager] Created directory: ${elementDir}`);\n    }\n    \n    // Create special directories for stateful elements\n    const agentStateDir = path.join(this.baseDir, ElementType.AGENT, '.state');\n    await fs.mkdir(agentStateDir, { recursive: true });\n    \n    logger.info('[PortfolioManager] Portfolio directory structure initialized');\n    \n    // Migration for v1.4.2 users: rename singular directories to plural\n    await this.migrateFromSingularDirectories();\n    \n    // Populate with default elements if this is a new installation\n    // Skip during tests to avoid interference\n    if (process.env.NODE_ENV !== 'test') {\n      try {\n        const defaultProvider = new DefaultElementProvider();\n        await defaultProvider.populateDefaults(this.baseDir);\n      } catch (error) {\n        logger.error('[PortfolioManager] Error populating default elements:', error);\n        // Log to stderr for Claude Desktop visibility\n        logger.error(`[PortfolioManager] CRITICAL: Failed to populate default elements: ${error instanceof Error ? error.message : String(error)}`);\n        // Continue anyway - empty portfolio is valid\n      }\n    }\n  }\n  \n  /**\n   * Check if portfolio directory exists\n   */\n  public async exists(): Promise<boolean> {\n    try {\n      await fs.access(this.baseDir);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Check if a filename appears to be a test element\n   * SAFETY: Pattern-based filtering only, no content parsing\n   */\n  public isTestElement(filename: string): boolean {\n    // Dangerous test patterns that should never appear in production\n    const dangerousPatterns = [\n      /^bin-sh/i,\n      /^rm-rf/i,\n      /^nc-e-bin/i,\n      /^python-c-import/i,\n      /^curl.*evil/i,\n      /^wget.*malicious/i,\n      /^eval-/i,\n      /^exec-/i,\n      /^bash-c-/i,\n      /^sh-c-/i,\n      /^powershell-/i,\n      /^cmd-c-/i,\n      /shell-injection/i\n    ];\n    \n    // Common test patterns\n    const testPatterns = [\n      /^test-/i,\n      /^memory-test-/i,\n      /^yaml-test/i,\n      /^perf-test-/i,\n      /^stability-test-/i,\n      /^roundtrip-test/i,\n      /test-persona/i,\n      /test-skill/i,\n      /test-template/i,\n      /test-agent/i,\n      /\\.test\\./,\n      /__test__/,\n      /test-data/,\n      /penetration-test/i,\n      /metadata-test/i,\n      /testpersona\\d+/i  // Generated test personas with timestamps\n    ];\n    \n    // Check dangerous patterns first\n    if (dangerousPatterns.some(pattern => pattern.test(filename))) {\n      logger.warn(`[PortfolioManager] Filtered dangerous test element: ${filename}`);\n      return true;\n    }\n    \n    // Check common test patterns\n    return testPatterns.some(pattern => pattern.test(filename));\n  }\n\n  /**\n   * List all elements of a specific type\n   */\n  public async listElements(type: ElementType): Promise<string[]> {\n    const elementDir = this.getElementDir(type);\n    const fileExtension = ELEMENT_FILE_EXTENSIONS[type] || DEFAULT_ELEMENT_FILE_EXTENSION;\n\n    try {\n      const files = await fs.readdir(elementDir);\n      // Filter for correct file extension based on element type and exclude test elements\n      return files\n        .filter(file => file.endsWith(fileExtension))\n        .filter(file => !this.isTestElement(file));\n    } catch (error) {\n      const err = error as NodeJS.ErrnoException;\n      \n      if (err.code === 'ENOENT') {\n        // Directory doesn't exist yet - this is expected for new installations\n        logger.debug(`[PortfolioManager] Element directory doesn't exist yet: ${elementDir}`);\n        return [];\n      }\n      \n      if (err.code === 'EACCES' || err.code === 'EPERM') {\n        // Permission denied - log but return empty array\n        ErrorHandler.logError('PortfolioManager.listElements', error, { elementDir });\n        return [];\n      }\n      \n      if (err.code === 'ENOTDIR') {\n        // Path exists but is not a directory\n        ErrorHandler.logError('PortfolioManager.listElements', error, { elementDir });\n        throw ErrorHandler.createError(`Path is not a directory: ${elementDir}`, ErrorCategory.SYSTEM_ERROR);\n      }\n      \n      // For any other errors, throw with context\n      ErrorHandler.logError('PortfolioManager.listElements', error, { elementDir });\n      throw ErrorHandler.wrapError(error, 'Failed to list elements', ErrorCategory.SYSTEM_ERROR);\n    }\n  }\n  \n  /**\n   * Get full path to an element file\n   */\n  public getElementPath(type: ElementType, filename: string): string {\n    // Validate filename to prevent path traversal\n    if (!filename || typeof filename !== 'string') {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'MEDIUM',\n        source: 'PortfolioManager.getElementPath',\n        details: `Invalid filename provided: ${typeof filename}`,\n        additionalData: { elementType: type, filename: String(filename) }\n      });\n      throw new Error('Invalid filename: must be a non-empty string');\n    }\n    \n    // Check for path traversal attempts\n    if (filename.includes('..') || filename.includes('/') || filename.includes('\\\\') || path.isAbsolute(filename)) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'HIGH',\n        source: 'PortfolioManager.getElementPath',\n        details: `Path traversal attempt detected in filename: ${filename}`,\n        additionalData: { elementType: type, filename }\n      });\n      throw new Error(`Invalid filename: contains path traversal characters: ${filename}`);\n    }\n    \n    // Additional validation for hidden files and special characters\n    if (filename.startsWith('.') || filename.includes('\\0')) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'MEDIUM',\n        source: 'PortfolioManager.getElementPath',\n        details: `Invalid filename characters detected: ${filename}`,\n        additionalData: { elementType: type, filename, hasHiddenFile: filename.startsWith('.'), hasNullByte: filename.includes('\\0') }\n      });\n      throw new Error(`Invalid filename: contains invalid characters: ${filename}`);\n    }\n    \n    // Ensure filename ends with .md\n    const safeFilename = filename.endsWith('.md') ? filename : `${filename}.md`;\n    return path.join(this.getElementDir(type), safeFilename);\n  }\n  \n  /**\n   * Check if an element exists\n   */\n  public async elementExists(type: ElementType, filename: string): Promise<boolean> {\n    try {\n      await fs.access(this.getElementPath(type, filename));\n      return true;\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Get legacy personas directory path (for migration)\n   */\n  public getLegacyPersonasDir(): string {\n    return path.join(homedir(), '.dollhouse', 'personas');\n  }\n  \n  /**\n   * Check if legacy personas directory exists\n   */\n  public async hasLegacyPersonas(): Promise<boolean> {\n    try {\n      await fs.access(this.getLegacyPersonasDir());\n      const files = await fs.readdir(this.getLegacyPersonasDir());\n      return files.some(file => file.endsWith('.md'));\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Get portfolio statistics\n   */\n  public async getStatistics(): Promise<Record<ElementType, number>> {\n    const stats: Record<string, number> = {};\n    \n    for (const elementType of Object.values(ElementType)) {\n      const elements = await this.listElements(elementType);\n      stats[elementType] = elements.length;\n    }\n    \n    return stats as Record<ElementType, number>;\n  }\n  \n  /**\n   * Migrate from v1.4.2 singular directory names to v1.4.3 plural names\n   * This handles the upgrade path for existing users\n   */\n  private async migrateFromSingularDirectories(): Promise<void> {\n    const oldToNew: Record<string, string> = {\n      'persona': 'personas',\n      'skill': 'skills',\n      'template': 'templates',\n      'agent': 'agents',\n      'memory': 'memories',\n      'ensemble': 'ensembles'\n    };\n    \n    for (const [oldName, newName] of Object.entries(oldToNew)) {\n      // Unicode normalize the directory names (even though they're hardcoded, for security audit)\n      const normalizedOld = UnicodeValidator.normalize(oldName);\n      const normalizedNew = UnicodeValidator.normalize(newName);\n      \n      if (!normalizedOld.isValid || !normalizedNew.isValid) {\n        // This should never happen with our hardcoded values, but for completeness\n        logger.error(`[PortfolioManager] Invalid Unicode in directory names during migration`);\n        continue;\n      }\n      \n      const oldDir = path.join(this.baseDir, normalizedOld.normalizedContent);\n      const newDir = path.join(this.baseDir, normalizedNew.normalizedContent);\n      \n      try {\n        // Check if old directory exists\n        await fs.access(oldDir);\n        \n        // Check if new directory already has content\n        try {\n          const newDirFiles = await fs.readdir(newDir);\n          if (newDirFiles.length > 0) {\n            logger.warn(\n              `[PortfolioManager] Both ${oldName} and ${newName} directories exist. Keeping ${newName}, skipping migration.`,\n              { oldDir, newDir, fileCount: newDirFiles.length }\n            );\n            continue;\n          }\n        } catch {\n          // New directory doesn't exist or is empty, proceed with migration\n        }\n        \n        // Perform the migration\n        logger.info(`[PortfolioManager] Migrating ${oldName} → ${newName}`);\n        await fs.rename(oldDir, newDir);\n        \n        // Log security event for audit trail\n        SecurityMonitor.logSecurityEvent({\n          type: 'DIRECTORY_MIGRATION',\n          severity: 'LOW',\n          source: 'PortfolioManager.migrateFromSingularDirectories',\n          details: `Migrated directory from ${oldName} to ${newName} for v1.4.3 compatibility`,\n          metadata: { oldDir, newDir }\n        });\n        \n      } catch (error) {\n        // Old directory doesn't exist, which is fine\n        if ((error as any).code !== 'ENOENT') {\n          logger.error(`[PortfolioManager] Error during migration of ${oldName}:`, error);\n        }\n      }\n    }\n  }\n}"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dollhousemcp/mcp-server",
3
- "version": "1.9.6",
3
+ "version": "1.9.7",
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",