@dollhousemcp/mcp-server 1.5.2 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/CHANGELOG.md +56 -0
  2. package/README.md +494 -111
  3. package/data/agents/code-reviewer.md +8 -1
  4. package/data/agents/research-assistant.md +8 -1
  5. package/data/agents/task-manager.md +8 -1
  6. package/data/ensembles/business-advisor.md +8 -1
  7. package/data/ensembles/creative-studio.md +8 -1
  8. package/data/ensembles/development-team.md +8 -1
  9. package/data/ensembles/security-analysis-team.md +8 -1
  10. package/data/memories/conversation-history.md +8 -1
  11. package/data/memories/learning-progress.md +8 -1
  12. package/data/memories/project-context.md +8 -1
  13. package/data/personas/business-consultant.md +8 -1
  14. package/data/personas/creative-writer.md +8 -1
  15. package/data/personas/debug-detective.md +8 -1
  16. package/data/personas/eli5-explainer.md +8 -1
  17. package/data/personas/security-analyst.md +8 -1
  18. package/data/personas/technical-analyst.md +8 -1
  19. package/data/skills/code-review.md +8 -1
  20. package/data/skills/creative-writing.md +8 -1
  21. package/data/skills/data-analysis.md +8 -1
  22. package/data/skills/penetration-testing.md +8 -1
  23. package/data/skills/research.md +8 -1
  24. package/data/skills/threat-modeling.md +8 -1
  25. package/data/skills/translation.md +8 -1
  26. package/data/templates/code-documentation.md +8 -1
  27. package/data/templates/email-professional.md +8 -1
  28. package/data/templates/meeting-notes.md +8 -1
  29. package/data/templates/penetration-test-report.md +8 -1
  30. package/data/templates/project-brief.md +8 -1
  31. package/data/templates/report-executive.md +8 -1
  32. package/data/templates/security-vulnerability-report.md +8 -1
  33. package/data/templates/threat-assessment-report.md +8 -1
  34. package/dist/auth/GitHubAuthManager.d.ts +6 -1
  35. package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
  36. package/dist/auth/GitHubAuthManager.js +45 -18
  37. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
  38. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
  39. package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
  40. package/dist/cache/CollectionCache.d.ts.map +1 -1
  41. package/dist/cache/CollectionCache.js +13 -3
  42. package/dist/cache/CollectionIndexCache.d.ts +77 -0
  43. package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
  44. package/dist/cache/CollectionIndexCache.js +349 -0
  45. package/dist/cache/LRUCache.d.ts +93 -0
  46. package/dist/cache/LRUCache.d.ts.map +1 -0
  47. package/dist/cache/LRUCache.js +299 -0
  48. package/dist/cache/index.d.ts +1 -0
  49. package/dist/cache/index.d.ts.map +1 -1
  50. package/dist/cache/index.js +2 -1
  51. package/dist/collection/CollectionBrowser.d.ts +21 -1
  52. package/dist/collection/CollectionBrowser.d.ts.map +1 -1
  53. package/dist/collection/CollectionBrowser.js +130 -10
  54. package/dist/collection/CollectionIndexManager.d.ts +151 -0
  55. package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
  56. package/dist/collection/CollectionIndexManager.js +499 -0
  57. package/dist/collection/CollectionSearch.d.ts +55 -0
  58. package/dist/collection/CollectionSearch.d.ts.map +1 -1
  59. package/dist/collection/CollectionSearch.js +338 -13
  60. package/dist/collection/CollectionSeeder.d.ts.map +1 -1
  61. package/dist/collection/CollectionSeeder.js +38 -1
  62. package/dist/collection/ElementInstaller.d.ts +31 -0
  63. package/dist/collection/ElementInstaller.d.ts.map +1 -1
  64. package/dist/collection/ElementInstaller.js +77 -15
  65. package/dist/collection/PersonaSubmitter.d.ts +1 -1
  66. package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
  67. package/dist/collection/PersonaSubmitter.js +2 -2
  68. package/dist/collection/index.d.ts +1 -0
  69. package/dist/collection/index.d.ts.map +1 -1
  70. package/dist/collection/index.js +2 -1
  71. package/dist/config/ConfigManager.d.ts +78 -0
  72. package/dist/config/ConfigManager.d.ts.map +1 -0
  73. package/dist/config/ConfigManager.js +216 -0
  74. package/dist/config/element-types.d.ts +135 -0
  75. package/dist/config/element-types.d.ts.map +1 -0
  76. package/dist/config/element-types.js +108 -0
  77. package/dist/config/index.d.ts +2 -0
  78. package/dist/config/index.d.ts.map +1 -1
  79. package/dist/config/index.js +3 -1
  80. package/dist/config/portfolio-constants.d.ts +83 -0
  81. package/dist/config/portfolio-constants.d.ts.map +1 -0
  82. package/dist/config/portfolio-constants.js +99 -0
  83. package/dist/elements/BaseElement.d.ts +14 -2
  84. package/dist/elements/BaseElement.d.ts.map +1 -1
  85. package/dist/elements/BaseElement.js +88 -6
  86. package/dist/elements/agents/Agent.d.ts +10 -1
  87. package/dist/elements/agents/Agent.d.ts.map +1 -1
  88. package/dist/elements/agents/Agent.js +66 -19
  89. package/dist/elements/agents/AgentManager.d.ts +2 -0
  90. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  91. package/dist/elements/agents/AgentManager.js +12 -10
  92. package/dist/elements/skills/Skill.d.ts +10 -1
  93. package/dist/elements/skills/Skill.d.ts.map +1 -1
  94. package/dist/elements/skills/Skill.js +40 -3
  95. package/dist/elements/skills/SkillManager.d.ts +1 -0
  96. package/dist/elements/skills/SkillManager.d.ts.map +1 -1
  97. package/dist/elements/skills/SkillManager.js +10 -4
  98. package/dist/elements/templates/Template.d.ts +10 -1
  99. package/dist/elements/templates/Template.d.ts.map +1 -1
  100. package/dist/elements/templates/Template.js +35 -18
  101. package/dist/elements/templates/TemplateManager.d.ts +1 -1
  102. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  103. package/dist/elements/templates/TemplateManager.js +6 -5
  104. package/dist/generated/version.d.ts +2 -2
  105. package/dist/generated/version.js +3 -3
  106. package/dist/index.barrel.d.ts +1 -2
  107. package/dist/index.barrel.d.ts.map +1 -1
  108. package/dist/index.barrel.js +2 -4
  109. package/dist/index.d.ts +143 -25
  110. package/dist/index.d.ts.map +1 -1
  111. package/dist/index.js +1883 -310
  112. package/dist/persona/PersonaElement.d.ts +10 -0
  113. package/dist/persona/PersonaElement.d.ts.map +1 -1
  114. package/dist/persona/PersonaElement.js +55 -32
  115. package/dist/persona/PersonaElementManager.d.ts.map +1 -1
  116. package/dist/persona/PersonaElementManager.js +13 -11
  117. package/dist/persona/PersonaLoader.d.ts.map +1 -1
  118. package/dist/persona/PersonaLoader.js +8 -2
  119. package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
  120. package/dist/persona/export-import/PersonaImporter.js +24 -5
  121. package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
  122. package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
  123. package/dist/persona/export-import/PersonaSharer.js +198 -22
  124. package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
  125. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  126. package/dist/portfolio/DefaultElementProvider.js +499 -7
  127. package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
  128. package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
  129. package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
  130. package/dist/portfolio/MigrationManager.d.ts.map +1 -1
  131. package/dist/portfolio/MigrationManager.js +136 -3
  132. package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
  133. package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
  134. package/dist/portfolio/PortfolioIndexManager.js +478 -0
  135. package/dist/portfolio/PortfolioManager.d.ts +5 -0
  136. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  137. package/dist/portfolio/PortfolioManager.js +61 -20
  138. package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
  139. package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
  140. package/dist/portfolio/PortfolioRepoManager.js +337 -0
  141. package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
  142. package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
  143. package/dist/portfolio/UnifiedIndexManager.js +1434 -0
  144. package/dist/portfolio/index.d.ts +15 -0
  145. package/dist/portfolio/index.d.ts.map +1 -0
  146. package/dist/portfolio/index.js +15 -0
  147. package/dist/portfolio/types.d.ts +7 -0
  148. package/dist/portfolio/types.d.ts.map +1 -1
  149. package/dist/portfolio/types.js +6 -1
  150. package/dist/security/InputValidator.d.ts.map +1 -1
  151. package/dist/security/InputValidator.js +50 -48
  152. package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
  153. package/dist/security/audit/SecurityAuditor.js +17 -9
  154. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  155. package/dist/security/audit/config/suppressions.js +19 -3
  156. package/dist/security/contentValidator.d.ts +2 -0
  157. package/dist/security/contentValidator.d.ts.map +1 -1
  158. package/dist/security/contentValidator.js +115 -4
  159. package/dist/security/secureYamlParser.d.ts +1 -0
  160. package/dist/security/secureYamlParser.d.ts.map +1 -1
  161. package/dist/security/secureYamlParser.js +29 -7
  162. package/dist/security/securityMonitor.d.ts +1 -1
  163. package/dist/security/securityMonitor.d.ts.map +1 -1
  164. package/dist/security/securityMonitor.js +1 -1
  165. package/dist/security/tokenManager.d.ts +1 -1
  166. package/dist/security/tokenManager.d.ts.map +1 -1
  167. package/dist/security/tokenManager.js +30 -10
  168. package/dist/server/ServerSetup.d.ts +22 -2
  169. package/dist/server/ServerSetup.d.ts.map +1 -1
  170. package/dist/server/ServerSetup.js +77 -12
  171. package/dist/server/tools/AuthTools.d.ts.map +1 -1
  172. package/dist/server/tools/AuthTools.js +33 -1
  173. package/dist/server/tools/BuildInfoTools.d.ts +25 -0
  174. package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
  175. package/dist/server/tools/BuildInfoTools.js +36 -0
  176. package/dist/server/tools/CollectionTools.d.ts.map +1 -1
  177. package/dist/server/tools/CollectionTools.js +55 -46
  178. package/dist/server/tools/ConfigTools.d.ts.map +1 -1
  179. package/dist/server/tools/ConfigTools.js +29 -1
  180. package/dist/server/tools/PersonaTools.d.ts +4 -2
  181. package/dist/server/tools/PersonaTools.d.ts.map +1 -1
  182. package/dist/server/tools/PersonaTools.js +5 -152
  183. package/dist/server/tools/PortfolioTools.d.ts +12 -0
  184. package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
  185. package/dist/server/tools/PortfolioTools.js +221 -0
  186. package/dist/server/tools/index.d.ts +3 -1
  187. package/dist/server/tools/index.d.ts.map +1 -1
  188. package/dist/server/tools/index.js +4 -2
  189. package/dist/server/types.d.ts +40 -5
  190. package/dist/server/types.d.ts.map +1 -1
  191. package/dist/server/types.js +1 -1
  192. package/dist/services/BuildInfoService.d.ts +84 -0
  193. package/dist/services/BuildInfoService.d.ts.map +1 -0
  194. package/dist/services/BuildInfoService.js +271 -0
  195. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
  196. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
  197. package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
  198. package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
  199. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
  200. package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
  201. package/dist/tools/portfolio/types.d.ts +41 -0
  202. package/dist/tools/portfolio/types.d.ts.map +1 -0
  203. package/dist/tools/portfolio/types.js +15 -0
  204. package/dist/types/collection.d.ts +51 -0
  205. package/dist/types/collection.d.ts.map +1 -1
  206. package/dist/types/collection.js +1 -1
  207. package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
  208. package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
  209. package/dist/utils/EarlyTerminationSearch.js +164 -0
  210. package/dist/utils/ErrorHandler.d.ts +86 -0
  211. package/dist/utils/ErrorHandler.d.ts.map +1 -0
  212. package/dist/utils/ErrorHandler.js +201 -0
  213. package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
  214. package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
  215. package/dist/utils/FileDiscoveryUtil.js +169 -0
  216. package/dist/utils/GitHubRateLimiter.d.ts +88 -0
  217. package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
  218. package/dist/utils/GitHubRateLimiter.js +315 -0
  219. package/dist/utils/PerformanceMonitor.d.ts +134 -0
  220. package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
  221. package/dist/utils/PerformanceMonitor.js +347 -0
  222. package/dist/utils/RateLimiter.d.ts.map +1 -0
  223. package/dist/utils/RateLimiter.js +172 -0
  224. package/dist/utils/SecureDownloader.d.ts +241 -0
  225. package/dist/utils/SecureDownloader.d.ts.map +1 -0
  226. package/dist/utils/SecureDownloader.js +759 -0
  227. package/dist/utils/ToolCache.d.ts +82 -0
  228. package/dist/utils/ToolCache.d.ts.map +1 -0
  229. package/dist/utils/ToolCache.js +196 -0
  230. package/dist/utils/errorCodes.d.ts +136 -0
  231. package/dist/utils/errorCodes.d.ts.map +1 -0
  232. package/dist/utils/errorCodes.js +87 -0
  233. package/dist/utils/index.d.ts +3 -0
  234. package/dist/utils/index.d.ts.map +1 -1
  235. package/dist/utils/index.js +4 -1
  236. package/dist/utils/installation.d.ts +1 -1
  237. package/dist/utils/installation.d.ts.map +1 -1
  238. package/dist/utils/installation.js +9 -8
  239. package/dist/utils/searchUtils.d.ts +31 -0
  240. package/dist/utils/searchUtils.d.ts.map +1 -1
  241. package/dist/utils/searchUtils.js +62 -1
  242. package/package.json +17 -7
  243. package/dist/config/updateConfig.d.ts +0 -84
  244. package/dist/config/updateConfig.d.ts.map +0 -1
  245. package/dist/config/updateConfig.js +0 -148
  246. package/dist/server/tools/UpdateTools.d.ts +0 -10
  247. package/dist/server/tools/UpdateTools.d.ts.map +0 -1
  248. package/dist/server/tools/UpdateTools.js +0 -85
  249. package/dist/update/BackupManager.d.ts +0 -63
  250. package/dist/update/BackupManager.d.ts.map +0 -1
  251. package/dist/update/BackupManager.js +0 -370
  252. package/dist/update/DependencyChecker.d.ts +0 -41
  253. package/dist/update/DependencyChecker.d.ts.map +0 -1
  254. package/dist/update/DependencyChecker.js +0 -132
  255. package/dist/update/RateLimiter.d.ts.map +0 -1
  256. package/dist/update/RateLimiter.js +0 -172
  257. package/dist/update/SignatureVerifier.d.ts +0 -71
  258. package/dist/update/SignatureVerifier.d.ts.map +0 -1
  259. package/dist/update/SignatureVerifier.js +0 -214
  260. package/dist/update/UpdateChecker.d.ts +0 -132
  261. package/dist/update/UpdateChecker.d.ts.map +0 -1
  262. package/dist/update/UpdateChecker.js +0 -506
  263. package/dist/update/UpdateManager.d.ts +0 -60
  264. package/dist/update/UpdateManager.d.ts.map +0 -1
  265. package/dist/update/UpdateManager.js +0 -730
  266. package/dist/update/VersionManager.d.ts +0 -31
  267. package/dist/update/VersionManager.d.ts.map +0 -1
  268. package/dist/update/VersionManager.js +0 -181
  269. package/dist/update/index.d.ts +0 -9
  270. package/dist/update/index.d.ts.map +0 -1
  271. package/dist/update/index.js +0 -9
  272. /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
@@ -12,8 +12,18 @@ export class CollectionCache {
12
12
  cacheDir;
13
13
  cacheFile;
14
14
  CACHE_TTL_MS = 24 * 60 * 60 * 1000; // 24 hours for collection cache
15
- constructor(baseDir = process.cwd()) {
16
- this.cacheDir = path.join(baseDir, '.dollhousemcp', 'cache');
15
+ constructor(baseDir) {
16
+ // Use environment variable if set, otherwise fall back to parameter or default
17
+ const envCacheDir = process.env.DOLLHOUSE_CACHE_DIR;
18
+ if (envCacheDir) {
19
+ this.cacheDir = envCacheDir;
20
+ logger.debug(`CollectionCache: Using environment cache directory: ${this.cacheDir}`);
21
+ }
22
+ else {
23
+ const defaultBaseDir = baseDir || process.cwd();
24
+ this.cacheDir = path.join(defaultBaseDir, '.dollhousemcp', 'cache');
25
+ logger.debug(`CollectionCache: Using default cache directory: ${this.cacheDir}`);
26
+ }
17
27
  this.cacheFile = path.join(this.cacheDir, 'collection-cache.json');
18
28
  }
19
29
  /**
@@ -159,4 +169,4 @@ export class CollectionCache {
159
169
  };
160
170
  }
161
171
  }
162
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CollectionCache.js","sourceRoot":"","sources":["../../src/cache/CollectionCache.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Persistent cache for collection data to support offline/anonymous browsing\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { logger } from '../utils/logger.js';\nimport { PathValidator } from '../security/pathValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\n\nexport interface CollectionItem {\n  name: string;\n  path: string;\n  sha: string;\n  content?: string;\n  last_modified?: string;\n}\n\nexport interface CollectionCacheEntry {\n  items: CollectionItem[];\n  timestamp: number;\n  etag?: string;\n}\n\n/**\n * Persistent cache for collection data that supports offline browsing\n */\nexport class CollectionCache {\n  private cacheDir: string;\n  private cacheFile: string;\n  private readonly CACHE_TTL_MS = 24 * 60 * 60 * 1000; // 24 hours for collection cache\n  \n  constructor(baseDir: string = process.cwd()) {\n    this.cacheDir = path.join(baseDir, '.dollhousemcp', 'cache');\n    this.cacheFile = path.join(this.cacheDir, 'collection-cache.json');\n  }\n  \n  /**\n   * Initialize cache directory\n   */\n  private async ensureCacheDir(): Promise<void> {\n    try {\n      await fs.mkdir(this.cacheDir, { recursive: true });\n    } catch (error) {\n      logger.error(`Failed to create cache directory: ${error}`);\n      throw error;\n    }\n  }\n  \n  /**\n   * Load collection data from persistent cache\n   */\n  async loadCache(): Promise<CollectionCacheEntry | null> {\n    try {\n      // Validate cache file path (basic security check)\n      if (this.cacheFile.includes('..') || this.cacheFile.includes('\\0')) {\n        // SECURITY FIX: Add audit logging for path traversal attempt detection\n        SecurityMonitor.logSecurityEvent({\n          type: 'PATH_TRAVERSAL_ATTEMPT',\n          severity: 'HIGH',\n          source: 'CollectionCache.loadCache',\n          details: `Potential path traversal attempt detected in cache file path: ${this.cacheFile.substring(0, 100)}`\n        });\n        logger.warn('Invalid cache file path, skipping cache load');\n        return null;\n      }\n      \n      const data = await fs.readFile(this.cacheFile, 'utf8');\n      const cache: CollectionCacheEntry = JSON.parse(data);\n      \n      // Check if cache is expired\n      if (Date.now() - cache.timestamp > this.CACHE_TTL_MS) {\n        logger.debug('Collection cache expired, will refresh from GitHub');\n        return null;\n      }\n      \n      logger.debug(`Loaded ${cache.items.length} items from collection cache`);\n      return cache;\n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug(`Failed to load collection cache: ${error}`);\n      }\n      return null;\n    }\n  }\n  \n  /**\n   * Save collection data to persistent cache\n   */\n  async saveCache(items: CollectionItem[], etag?: string): Promise<void> {\n    try {\n      await this.ensureCacheDir();\n      \n      const cacheEntry: CollectionCacheEntry = {\n        items,\n        timestamp: Date.now(),\n        etag\n      };\n      \n      const data = JSON.stringify(cacheEntry, null, 2);\n      await fs.writeFile(this.cacheFile, data, 'utf8');\n      \n      logger.debug(`Saved ${items.length} items to collection cache`);\n      \n      // SECURITY FIX: Add audit logging for cache write operations\n      logger.debug('Security audit: Cache write operation completed successfully');\n      \n      // Log operation completed successfully\n      logger.debug(`Cache file operation completed with ${items.length} items`);\n    } catch (error) {\n      logger.error(`Failed to save collection cache: ${error}`);\n      // Don't throw - caching failures shouldn't break functionality\n    }\n  }\n  \n  /**\n   * Search cached collection items with fuzzy matching\n   */\n  async searchCache(query: string): Promise<CollectionItem[]> {\n    const cache = await this.loadCache();\n    if (!cache) {\n      return [];\n    }\n    \n    const normalizedQuery = this.normalizeSearchTerm(query);\n    return cache.items.filter(item => {\n      // Search in filename and path with normalization\n      const normalizedName = this.normalizeSearchTerm(item.name);\n      const normalizedPath = this.normalizeSearchTerm(item.path);\n      \n      return normalizedName.includes(normalizedQuery) || \n             normalizedPath.includes(normalizedQuery) ||\n             (item.content && this.normalizeSearchTerm(item.content).includes(normalizedQuery));\n    });\n  }\n  \n  /**\n   * Normalize search terms for better matching (handles spaces, dashes, etc.)\n   */\n  private normalizeSearchTerm(term: string): string {\n    return term.toLowerCase()\n      .replace(/[-_\\s]+/g, ' ')  // Convert dashes, underscores to spaces\n      .replace(/\\.md$/, '')       // Remove .md extension\n      .trim();\n  }\n  \n  /**\n   * Get cached collection items by type/path\n   */\n  async getItemsByPath(pathPrefix: string): Promise<CollectionItem[]> {\n    const cache = await this.loadCache();\n    if (!cache) {\n      return [];\n    }\n    \n    return cache.items.filter(item => item.path.startsWith(pathPrefix));\n  }\n  \n  /**\n   * Check if cache exists and is valid\n   */\n  async isCacheValid(): Promise<boolean> {\n    const cache = await this.loadCache();\n    return cache !== null;\n  }\n  \n  /**\n   * Clear the cache\n   */\n  async clearCache(): Promise<void> {\n    try {\n      await fs.unlink(this.cacheFile);\n      logger.debug('Collection cache cleared');\n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug(`Failed to clear collection cache: ${error}`);\n      }\n    }\n  }\n  \n  /**\n   * Get cache stats for debugging\n   */\n  async getCacheStats(): Promise<{ itemCount: number; cacheAge: number; isValid: boolean }> {\n    const cache = await this.loadCache();\n    if (!cache) {\n      return { itemCount: 0, cacheAge: 0, isValid: false };\n    }\n    \n    return {\n      itemCount: cache.items.length,\n      cacheAge: Date.now() - cache.timestamp,\n      isValid: Date.now() - cache.timestamp <= this.CACHE_TTL_MS\n    };\n  }\n}"]}
172
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CollectionCache.js","sourceRoot":"","sources":["../../src/cache/CollectionCache.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Persistent cache for collection data to support offline/anonymous browsing\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { logger } from '../utils/logger.js';\nimport { PathValidator } from '../security/pathValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\n\nexport interface CollectionItem {\n  name: string;\n  path: string;\n  sha: string;\n  content?: string;\n  last_modified?: string;\n}\n\nexport interface CollectionCacheEntry {\n  items: CollectionItem[];\n  timestamp: number;\n  etag?: string;\n}\n\n/**\n * Persistent cache for collection data that supports offline browsing\n */\nexport class CollectionCache {\n  private cacheDir: string;\n  private cacheFile: string;\n  private readonly CACHE_TTL_MS = 24 * 60 * 60 * 1000; // 24 hours for collection cache\n  \n  constructor(baseDir?: string) {\n    // Use environment variable if set, otherwise fall back to parameter or default\n    const envCacheDir = process.env.DOLLHOUSE_CACHE_DIR;\n    if (envCacheDir) {\n      this.cacheDir = envCacheDir;\n      logger.debug(`CollectionCache: Using environment cache directory: ${this.cacheDir}`);\n    } else {\n      const defaultBaseDir = baseDir || process.cwd();\n      this.cacheDir = path.join(defaultBaseDir, '.dollhousemcp', 'cache');\n      logger.debug(`CollectionCache: Using default cache directory: ${this.cacheDir}`);\n    }\n    this.cacheFile = path.join(this.cacheDir, 'collection-cache.json');\n  }\n  \n  /**\n   * Initialize cache directory\n   */\n  private async ensureCacheDir(): Promise<void> {\n    try {\n      await fs.mkdir(this.cacheDir, { recursive: true });\n    } catch (error) {\n      logger.error(`Failed to create cache directory: ${error}`);\n      throw error;\n    }\n  }\n  \n  /**\n   * Load collection data from persistent cache\n   */\n  async loadCache(): Promise<CollectionCacheEntry | null> {\n    try {\n      // Validate cache file path (basic security check)\n      if (this.cacheFile.includes('..') || this.cacheFile.includes('\\0')) {\n        // SECURITY FIX: Add audit logging for path traversal attempt detection\n        SecurityMonitor.logSecurityEvent({\n          type: 'PATH_TRAVERSAL_ATTEMPT',\n          severity: 'HIGH',\n          source: 'CollectionCache.loadCache',\n          details: `Potential path traversal attempt detected in cache file path: ${this.cacheFile.substring(0, 100)}`\n        });\n        logger.warn('Invalid cache file path, skipping cache load');\n        return null;\n      }\n      \n      const data = await fs.readFile(this.cacheFile, 'utf8');\n      const cache: CollectionCacheEntry = JSON.parse(data);\n      \n      // Check if cache is expired\n      if (Date.now() - cache.timestamp > this.CACHE_TTL_MS) {\n        logger.debug('Collection cache expired, will refresh from GitHub');\n        return null;\n      }\n      \n      logger.debug(`Loaded ${cache.items.length} items from collection cache`);\n      return cache;\n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug(`Failed to load collection cache: ${error}`);\n      }\n      return null;\n    }\n  }\n  \n  /**\n   * Save collection data to persistent cache\n   */\n  async saveCache(items: CollectionItem[], etag?: string): Promise<void> {\n    try {\n      await this.ensureCacheDir();\n      \n      const cacheEntry: CollectionCacheEntry = {\n        items,\n        timestamp: Date.now(),\n        etag\n      };\n      \n      const data = JSON.stringify(cacheEntry, null, 2);\n      await fs.writeFile(this.cacheFile, data, 'utf8');\n      \n      logger.debug(`Saved ${items.length} items to collection cache`);\n      \n      // SECURITY FIX: Add audit logging for cache write operations\n      logger.debug('Security audit: Cache write operation completed successfully');\n      \n      // Log operation completed successfully\n      logger.debug(`Cache file operation completed with ${items.length} items`);\n    } catch (error) {\n      logger.error(`Failed to save collection cache: ${error}`);\n      // Don't throw - caching failures shouldn't break functionality\n    }\n  }\n  \n  /**\n   * Search cached collection items with fuzzy matching\n   */\n  async searchCache(query: string): Promise<CollectionItem[]> {\n    const cache = await this.loadCache();\n    if (!cache) {\n      return [];\n    }\n    \n    const normalizedQuery = this.normalizeSearchTerm(query);\n    return cache.items.filter(item => {\n      // Search in filename and path with normalization\n      const normalizedName = this.normalizeSearchTerm(item.name);\n      const normalizedPath = this.normalizeSearchTerm(item.path);\n      \n      return normalizedName.includes(normalizedQuery) || \n             normalizedPath.includes(normalizedQuery) ||\n             (item.content && this.normalizeSearchTerm(item.content).includes(normalizedQuery));\n    });\n  }\n  \n  /**\n   * Normalize search terms for better matching (handles spaces, dashes, etc.)\n   */\n  private normalizeSearchTerm(term: string): string {\n    return term.toLowerCase()\n      .replace(/[-_\\s]+/g, ' ')  // Convert dashes, underscores to spaces\n      .replace(/\\.md$/, '')       // Remove .md extension\n      .trim();\n  }\n  \n  /**\n   * Get cached collection items by type/path\n   */\n  async getItemsByPath(pathPrefix: string): Promise<CollectionItem[]> {\n    const cache = await this.loadCache();\n    if (!cache) {\n      return [];\n    }\n    \n    return cache.items.filter(item => item.path.startsWith(pathPrefix));\n  }\n  \n  /**\n   * Check if cache exists and is valid\n   */\n  async isCacheValid(): Promise<boolean> {\n    const cache = await this.loadCache();\n    return cache !== null;\n  }\n  \n  /**\n   * Clear the cache\n   */\n  async clearCache(): Promise<void> {\n    try {\n      await fs.unlink(this.cacheFile);\n      logger.debug('Collection cache cleared');\n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug(`Failed to clear collection cache: ${error}`);\n      }\n    }\n  }\n  \n  /**\n   * Get cache stats for debugging\n   */\n  async getCacheStats(): Promise<{ itemCount: number; cacheAge: number; isValid: boolean }> {\n    const cache = await this.loadCache();\n    if (!cache) {\n      return { itemCount: 0, cacheAge: 0, isValid: false };\n    }\n    \n    return {\n      itemCount: cache.items.length,\n      cacheAge: Date.now() - cache.timestamp,\n      isValid: Date.now() - cache.timestamp <= this.CACHE_TTL_MS\n    };\n  }\n}"]}
@@ -0,0 +1,77 @@
1
+ /**
2
+ * Smart cache for collection index with conditional fetching and performance optimization
3
+ */
4
+ import { CollectionIndex } from '../types/collection.js';
5
+ import { GitHubClient } from '../collection/GitHubClient.js';
6
+ export declare class CollectionIndexCache {
7
+ private cache;
8
+ private readonly TTL;
9
+ private readonly INDEX_URL;
10
+ private cacheDir;
11
+ private cacheFile;
12
+ private githubClient;
13
+ private performanceMonitor;
14
+ private memoryCache;
15
+ private fetchPromise;
16
+ constructor(githubClient: GitHubClient, baseDir?: string);
17
+ /**
18
+ * Get the collection index with performance optimization and lazy loading
19
+ */
20
+ getIndex(lazyLoad?: boolean): Promise<CollectionIndex>;
21
+ /**
22
+ * Check if current cache is valid (within TTL)
23
+ */
24
+ private isValid;
25
+ /**
26
+ * Fetch fresh index from GitHub with conditional requests
27
+ */
28
+ private fetchFresh;
29
+ /**
30
+ * Validate the structure of a collection index
31
+ */
32
+ private validateIndexStructure;
33
+ /**
34
+ * Save cache to persistent storage
35
+ */
36
+ private saveToDisk;
37
+ /**
38
+ * Load cache from persistent storage
39
+ */
40
+ private loadFromDisk;
41
+ /**
42
+ * Ensure cache directory exists
43
+ */
44
+ private ensureCacheDir;
45
+ /**
46
+ * Clear all cache data with performance monitoring
47
+ */
48
+ clearCache(): Promise<void>;
49
+ /**
50
+ * Get comprehensive cache statistics for debugging and monitoring
51
+ */
52
+ getCacheStats(): {
53
+ isValid: boolean;
54
+ age: number;
55
+ hasCache: boolean;
56
+ elements: number;
57
+ memoryCache: any;
58
+ performanceMetrics: any;
59
+ };
60
+ /**
61
+ * Fetch fresh index with comprehensive fallback strategy
62
+ */
63
+ private fetchFreshWithFallback;
64
+ /**
65
+ * Record performance metrics for cache operations
66
+ */
67
+ private recordPerformanceMetrics;
68
+ /**
69
+ * Calculate cache hit rate (placeholder for future implementation)
70
+ */
71
+ private calculateCacheHitRate;
72
+ /**
73
+ * Calculate average access time (placeholder for future implementation)
74
+ */
75
+ private calculateAverageAccessTime;
76
+ }
77
+ //# sourceMappingURL=CollectionIndexCache.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"CollectionIndexCache.d.ts","sourceRoot":"","sources":["../../src/cache/CollectionIndexCache.ts"],"names":[],"mappings":"AAAA;;GAEG;AAIH,OAAO,EAAE,eAAe,EAAe,MAAM,wBAAwB,CAAC;AAEtE,OAAO,EAAE,YAAY,EAAE,MAAM,+BAA+B,CAAC;AAK7D,qBAAa,oBAAoB;IAC/B,OAAO,CAAC,KAAK,CAA4B;IACzC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAkB;IACtC,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAiG;IAC3H,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,YAAY,CAAe;IACnC,OAAO,CAAC,kBAAkB,CAAqB;IAC/C,OAAO,CAAC,WAAW,CAAgB;IACnC,OAAO,CAAC,YAAY,CAAyC;gBAEjD,YAAY,EAAE,YAAY,EAAE,OAAO,GAAE,MAAsB;IAiBvE;;OAEG;IACG,QAAQ,CAAC,QAAQ,GAAE,OAAe,GAAG,OAAO,CAAC,eAAe,CAAC;IAiEnE;;OAEG;IACH,OAAO,CAAC,OAAO;IASf;;OAEG;YACW,UAAU;IAiExB;;OAEG;IACH,OAAO,CAAC,sBAAsB;IAc9B;;OAEG;YACW,UAAU;IAmBxB;;OAEG;YACW,YAAY;IA4B1B;;OAEG;YACW,cAAc;IAS5B;;OAEG;IACG,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAwBjC;;OAEG;IACH,aAAa,IAAI;QACf,OAAO,EAAE,OAAO,CAAC;QACjB,GAAG,EAAE,MAAM,CAAC;QACZ,QAAQ,EAAE,OAAO,CAAC;QAClB,QAAQ,EAAE,MAAM,CAAC;QACjB,WAAW,EAAE,GAAG,CAAC;QACjB,kBAAkB,EAAE,GAAG,CAAC;KACzB;IA8BD;;OAEG;YACW,sBAAsB;IAuBpC;;OAEG;IACH,OAAO,CAAC,wBAAwB;IAqBhC;;OAEG;IACH,OAAO,CAAC,qBAAqB;IAK7B;;OAEG;IACH,OAAO,CAAC,0BAA0B;CAInC"}
@@ -0,0 +1,349 @@
1
+ /**
2
+ * Smart cache for collection index with conditional fetching and performance optimization
3
+ */
4
+ import * as fs from 'fs/promises';
5
+ import * as path from 'path';
6
+ import { logger } from '../utils/logger.js';
7
+ import { SecurityMonitor } from '../security/securityMonitor.js';
8
+ import { CacheFactory } from './LRUCache.js';
9
+ import { PerformanceMonitor } from '../utils/PerformanceMonitor.js';
10
+ export class CollectionIndexCache {
11
+ cache = null;
12
+ TTL = 15 * 60 * 1000; // 15 minutes in milliseconds
13
+ INDEX_URL = 'https://raw.githubusercontent.com/DollhouseMCP/collection/main/public/collection-index.json';
14
+ cacheDir;
15
+ cacheFile;
16
+ githubClient;
17
+ performanceMonitor;
18
+ memoryCache;
19
+ fetchPromise = null; // Prevent concurrent fetches
20
+ constructor(githubClient, baseDir = process.cwd()) {
21
+ this.githubClient = githubClient;
22
+ this.cacheDir = path.join(baseDir, '.dollhousemcp', 'cache');
23
+ this.cacheFile = path.join(this.cacheDir, 'collection-index-cache.json');
24
+ this.performanceMonitor = PerformanceMonitor.getInstance();
25
+ // Initialize memory cache for frequently accessed data
26
+ this.memoryCache = CacheFactory.createAPICache({
27
+ maxSize: 50,
28
+ maxMemoryMB: 10,
29
+ ttlMs: 5 * 60 * 1000, // 5 minutes
30
+ onEviction: (key, value) => {
31
+ logger.debug('Collection memory cache eviction', { key });
32
+ }
33
+ });
34
+ }
35
+ /**
36
+ * Get the collection index with performance optimization and lazy loading
37
+ */
38
+ async getIndex(lazyLoad = false) {
39
+ const startTime = Date.now();
40
+ const memoryBefore = process.memoryUsage().heapUsed;
41
+ try {
42
+ // Check memory cache first for fastest access
43
+ const memoryCached = this.memoryCache.get('collection-index');
44
+ if (memoryCached && this.isValid()) {
45
+ logger.debug('Using memory cached collection index');
46
+ this.recordPerformanceMetrics(startTime, memoryBefore, 'memory-hit');
47
+ return memoryCached;
48
+ }
49
+ // Check if we have valid disk cached data
50
+ if (this.isValid()) {
51
+ logger.debug('Using valid disk cached collection index');
52
+ const result = this.cache.data;
53
+ this.memoryCache.set('collection-index', result);
54
+ this.recordPerformanceMetrics(startTime, memoryBefore, 'disk-hit');
55
+ return result;
56
+ }
57
+ // Prevent concurrent fetches
58
+ if (this.fetchPromise) {
59
+ logger.debug('Waiting for ongoing collection index fetch');
60
+ return await this.fetchPromise;
61
+ }
62
+ // Lazy loading: Only fetch if not in lazy mode or absolutely necessary
63
+ if (lazyLoad && this.cache?.data) {
64
+ logger.debug('Using stale cache in lazy load mode');
65
+ this.recordPerformanceMetrics(startTime, memoryBefore, 'lazy-stale');
66
+ return this.cache.data;
67
+ }
68
+ // Create fetch promise to prevent concurrent requests
69
+ this.fetchPromise = this.fetchFreshWithFallback();
70
+ try {
71
+ const result = await this.fetchPromise;
72
+ this.memoryCache.set('collection-index', result);
73
+ this.recordPerformanceMetrics(startTime, memoryBefore, 'fresh-fetch');
74
+ return result;
75
+ }
76
+ finally {
77
+ this.fetchPromise = null;
78
+ }
79
+ }
80
+ catch (error) {
81
+ logger.error('Failed to get collection index:', error);
82
+ // Try loading from persistent cache as last resort
83
+ const persistentCache = await this.loadFromDisk();
84
+ if (persistentCache) {
85
+ logger.debug('Using persistent cache as last resort');
86
+ this.cache = persistentCache;
87
+ const result = persistentCache.data;
88
+ this.memoryCache.set('collection-index', result);
89
+ this.recordPerformanceMetrics(startTime, memoryBefore, 'disk-fallback');
90
+ return result;
91
+ }
92
+ throw error;
93
+ }
94
+ }
95
+ /**
96
+ * Check if current cache is valid (within TTL)
97
+ */
98
+ isValid() {
99
+ if (!this.cache) {
100
+ return false;
101
+ }
102
+ const age = Date.now() - this.cache.fetchedAt.getTime();
103
+ return age < this.TTL;
104
+ }
105
+ /**
106
+ * Fetch fresh index from GitHub with conditional requests
107
+ */
108
+ async fetchFresh() {
109
+ try {
110
+ // Build headers for conditional request
111
+ const headers = {
112
+ 'Accept': 'application/json',
113
+ 'User-Agent': 'DollhouseMCP/1.0'
114
+ };
115
+ // Add conditional headers if we have cache
116
+ if (this.cache?.etag) {
117
+ headers['If-None-Match'] = this.cache.etag;
118
+ }
119
+ if (this.cache?.lastModified) {
120
+ headers['If-Modified-Since'] = this.cache.lastModified;
121
+ }
122
+ // Use fetch directly for better control over conditional requests
123
+ const response = await fetch(this.INDEX_URL, { headers });
124
+ // 304 Not Modified - use cached data
125
+ if (response.status === 304) {
126
+ if (this.cache) {
127
+ // Update timestamp to extend cache validity
128
+ this.cache.fetchedAt = new Date();
129
+ await this.saveToDisk(this.cache);
130
+ logger.debug('Collection index not modified - refreshed cache timestamp');
131
+ return this.cache.data;
132
+ }
133
+ }
134
+ if (!response.ok) {
135
+ throw new Error(`HTTP ${response.status}: ${response.statusText}`);
136
+ }
137
+ const indexData = await response.json();
138
+ // Validate the index structure
139
+ if (!this.validateIndexStructure(indexData)) {
140
+ throw new Error('Invalid collection index structure received');
141
+ }
142
+ // Create new cache entry
143
+ const newCache = {
144
+ data: indexData,
145
+ fetchedAt: new Date(),
146
+ etag: response.headers.get('etag') || undefined,
147
+ lastModified: response.headers.get('last-modified') || undefined
148
+ };
149
+ this.cache = newCache;
150
+ // Save to persistent cache in background
151
+ this.saveToDisk(newCache).catch(error => {
152
+ logger.debug('Failed to save index cache to disk:', error);
153
+ });
154
+ logger.debug(`Fresh collection index fetched with ${indexData.total_elements} elements`);
155
+ return indexData;
156
+ }
157
+ catch (error) {
158
+ logger.debug('Failed to fetch fresh collection index:', error);
159
+ return null;
160
+ }
161
+ }
162
+ /**
163
+ * Validate the structure of a collection index
164
+ */
165
+ validateIndexStructure(index) {
166
+ return (index &&
167
+ typeof index === 'object' &&
168
+ typeof index.version === 'string' &&
169
+ typeof index.generated === 'string' &&
170
+ typeof index.total_elements === 'number' &&
171
+ index.index &&
172
+ typeof index.index === 'object' &&
173
+ index.metadata &&
174
+ typeof index.metadata === 'object');
175
+ }
176
+ /**
177
+ * Save cache to persistent storage
178
+ */
179
+ async saveToDisk(cache) {
180
+ try {
181
+ await this.ensureCacheDir();
182
+ const cacheData = {
183
+ ...cache,
184
+ fetchedAt: cache.fetchedAt.toISOString() // Serialize date
185
+ };
186
+ const data = JSON.stringify(cacheData, null, 2);
187
+ await fs.writeFile(this.cacheFile, data, 'utf8');
188
+ logger.debug('Collection index cache saved to disk');
189
+ }
190
+ catch (error) {
191
+ logger.debug('Failed to save collection index cache:', error);
192
+ // Don't throw - cache persistence failures shouldn't break functionality
193
+ }
194
+ }
195
+ /**
196
+ * Load cache from persistent storage
197
+ */
198
+ async loadFromDisk() {
199
+ try {
200
+ // Basic security check for path traversal
201
+ if (this.cacheFile.includes('..') || this.cacheFile.includes('\0')) {
202
+ SecurityMonitor.logSecurityEvent({
203
+ type: 'PATH_TRAVERSAL_ATTEMPT',
204
+ severity: 'HIGH',
205
+ source: 'CollectionIndexCache.loadFromDisk',
206
+ details: `Potential path traversal attempt detected: ${this.cacheFile.substring(0, 100)}`
207
+ });
208
+ return null;
209
+ }
210
+ const data = await fs.readFile(this.cacheFile, 'utf8');
211
+ const cacheData = JSON.parse(data);
212
+ return {
213
+ ...cacheData,
214
+ fetchedAt: new Date(cacheData.fetchedAt) // Deserialize date
215
+ };
216
+ }
217
+ catch (error) {
218
+ if (error.code !== 'ENOENT') {
219
+ logger.debug('Failed to load collection index cache from disk:', error);
220
+ }
221
+ return null;
222
+ }
223
+ }
224
+ /**
225
+ * Ensure cache directory exists
226
+ */
227
+ async ensureCacheDir() {
228
+ try {
229
+ await fs.mkdir(this.cacheDir, { recursive: true });
230
+ }
231
+ catch (error) {
232
+ logger.error('Failed to create cache directory:', error);
233
+ throw error;
234
+ }
235
+ }
236
+ /**
237
+ * Clear all cache data with performance monitoring
238
+ */
239
+ async clearCache() {
240
+ const startTime = Date.now();
241
+ this.cache = null;
242
+ this.memoryCache.clear();
243
+ // Cancel any ongoing fetch
244
+ this.fetchPromise = null;
245
+ try {
246
+ await fs.unlink(this.cacheFile);
247
+ logger.debug('Collection index cache cleared from disk');
248
+ }
249
+ catch (error) {
250
+ if (error.code !== 'ENOENT') {
251
+ logger.debug('Failed to clear collection index cache:', error);
252
+ }
253
+ }
254
+ logger.debug('Collection cache cleared', {
255
+ duration: Date.now() - startTime,
256
+ memoryFreed: this.memoryCache.getMemoryUsageMB()
257
+ });
258
+ }
259
+ /**
260
+ * Get comprehensive cache statistics for debugging and monitoring
261
+ */
262
+ getCacheStats() {
263
+ if (!this.cache) {
264
+ return {
265
+ isValid: false,
266
+ age: 0,
267
+ hasCache: false,
268
+ elements: 0,
269
+ memoryCache: this.memoryCache.getStats(),
270
+ performanceMetrics: null
271
+ };
272
+ }
273
+ const age = Date.now() - this.cache.fetchedAt.getTime();
274
+ return {
275
+ isValid: this.isValid(),
276
+ age,
277
+ hasCache: true,
278
+ elements: this.cache.data.total_elements,
279
+ memoryCache: this.memoryCache.getStats(),
280
+ performanceMetrics: {
281
+ cacheHitRate: this.calculateCacheHitRate(),
282
+ averageAccessTime: this.calculateAverageAccessTime()
283
+ }
284
+ };
285
+ }
286
+ // =====================================================
287
+ // PRIVATE HELPER METHODS FOR PERFORMANCE
288
+ // =====================================================
289
+ /**
290
+ * Fetch fresh index with comprehensive fallback strategy
291
+ */
292
+ async fetchFreshWithFallback() {
293
+ try {
294
+ // Try to fetch fresh index
295
+ const freshIndex = await this.fetchFresh();
296
+ if (freshIndex) {
297
+ logger.debug('Collection index fetched successfully');
298
+ return freshIndex;
299
+ }
300
+ }
301
+ catch (error) {
302
+ logger.warn('Fresh fetch failed, trying fallback', {
303
+ error: error instanceof Error ? error.message : String(error)
304
+ });
305
+ }
306
+ // Fall back to stale cache if available
307
+ if (this.cache?.data) {
308
+ logger.debug('Using stale cached collection index as fallback');
309
+ return this.cache.data;
310
+ }
311
+ throw new Error('No collection index available - fresh fetch failed and no cache exists');
312
+ }
313
+ /**
314
+ * Record performance metrics for cache operations
315
+ */
316
+ recordPerformanceMetrics(startTime, memoryBefore, operation) {
317
+ const duration = Date.now() - startTime;
318
+ const memoryAfter = process.memoryUsage().heapUsed;
319
+ logger.debug('Collection cache operation completed', {
320
+ operation,
321
+ duration,
322
+ memoryUsageMB: (memoryAfter - memoryBefore) / (1024 * 1024)
323
+ });
324
+ // Record cache performance metrics
325
+ this.performanceMonitor.recordCachePerformance('collectionIndex', {
326
+ hitRate: operation.includes('hit') ? 1 : 0,
327
+ avgHitTime: operation.includes('hit') ? duration : 0,
328
+ avgMissTime: operation.includes('hit') ? 0 : duration,
329
+ totalHits: operation.includes('hit') ? 1 : 0,
330
+ totalMisses: operation.includes('hit') ? 0 : 1,
331
+ evictions: 0
332
+ });
333
+ }
334
+ /**
335
+ * Calculate cache hit rate (placeholder for future implementation)
336
+ */
337
+ calculateCacheHitRate() {
338
+ // This would be implemented with actual metrics tracking
339
+ return this.memoryCache.getStats().hitRate;
340
+ }
341
+ /**
342
+ * Calculate average access time (placeholder for future implementation)
343
+ */
344
+ calculateAverageAccessTime() {
345
+ // This would be implemented with actual timing metrics
346
+ return 5; // Placeholder value
347
+ }
348
+ }
349
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CollectionIndexCache.js","sourceRoot":"","sources":["../../src/cache/CollectionIndexCache.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Smart cache for collection index with conditional fetching and performance optimization\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { CollectionIndex, CachedIndex } from '../types/collection.js';\nimport { logger } from '../utils/logger.js';\nimport { GitHubClient } from '../collection/GitHubClient.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { LRUCache, CacheFactory } from './LRUCache.js';\nimport { PerformanceMonitor } from '../utils/PerformanceMonitor.js';\n\nexport class CollectionIndexCache {\n  private cache: CachedIndex | null = null;\n  private readonly TTL = 15 * 60 * 1000; // 15 minutes in milliseconds\n  private readonly INDEX_URL = 'https://raw.githubusercontent.com/DollhouseMCP/collection/main/public/collection-index.json';\n  private cacheDir: string;\n  private cacheFile: string;\n  private githubClient: GitHubClient;\n  private performanceMonitor: PerformanceMonitor;\n  private memoryCache: LRUCache<any>;\n  private fetchPromise: Promise<CollectionIndex> | null = null; // Prevent concurrent fetches\n  \n  constructor(githubClient: GitHubClient, baseDir: string = process.cwd()) {\n    this.githubClient = githubClient;\n    this.cacheDir = path.join(baseDir, '.dollhousemcp', 'cache');\n    this.cacheFile = path.join(this.cacheDir, 'collection-index-cache.json');\n    this.performanceMonitor = PerformanceMonitor.getInstance();\n    \n    // Initialize memory cache for frequently accessed data\n    this.memoryCache = CacheFactory.createAPICache({\n      maxSize: 50,\n      maxMemoryMB: 10,\n      ttlMs: 5 * 60 * 1000, // 5 minutes\n      onEviction: (key, value) => {\n        logger.debug('Collection memory cache eviction', { key });\n      }\n    });\n  }\n  \n  /**\n   * Get the collection index with performance optimization and lazy loading\n   */\n  async getIndex(lazyLoad: boolean = false): Promise<CollectionIndex> {\n    const startTime = Date.now();\n    const memoryBefore = process.memoryUsage().heapUsed;\n    \n    try {\n      // Check memory cache first for fastest access\n      const memoryCached = this.memoryCache.get('collection-index');\n      if (memoryCached && this.isValid()) {\n        logger.debug('Using memory cached collection index');\n        this.recordPerformanceMetrics(startTime, memoryBefore, 'memory-hit');\n        return memoryCached;\n      }\n      \n      // Check if we have valid disk cached data\n      if (this.isValid()) {\n        logger.debug('Using valid disk cached collection index');\n        const result = this.cache!.data;\n        this.memoryCache.set('collection-index', result);\n        this.recordPerformanceMetrics(startTime, memoryBefore, 'disk-hit');\n        return result;\n      }\n      \n      // Prevent concurrent fetches\n      if (this.fetchPromise) {\n        logger.debug('Waiting for ongoing collection index fetch');\n        return await this.fetchPromise;\n      }\n      \n      // Lazy loading: Only fetch if not in lazy mode or absolutely necessary\n      if (lazyLoad && this.cache?.data) {\n        logger.debug('Using stale cache in lazy load mode');\n        this.recordPerformanceMetrics(startTime, memoryBefore, 'lazy-stale');\n        return this.cache.data;\n      }\n      \n      // Create fetch promise to prevent concurrent requests\n      this.fetchPromise = this.fetchFreshWithFallback();\n      \n      try {\n        const result = await this.fetchPromise;\n        this.memoryCache.set('collection-index', result);\n        this.recordPerformanceMetrics(startTime, memoryBefore, 'fresh-fetch');\n        return result;\n      } finally {\n        this.fetchPromise = null;\n      }\n      \n    } catch (error) {\n      logger.error('Failed to get collection index:', error);\n      \n      // Try loading from persistent cache as last resort\n      const persistentCache = await this.loadFromDisk();\n      if (persistentCache) {\n        logger.debug('Using persistent cache as last resort');\n        this.cache = persistentCache;\n        const result = persistentCache.data;\n        this.memoryCache.set('collection-index', result);\n        this.recordPerformanceMetrics(startTime, memoryBefore, 'disk-fallback');\n        return result;\n      }\n      \n      throw error;\n    }\n  }\n  \n  /**\n   * Check if current cache is valid (within TTL)\n   */\n  private isValid(): boolean {\n    if (!this.cache) {\n      return false;\n    }\n    \n    const age = Date.now() - this.cache.fetchedAt.getTime();\n    return age < this.TTL;\n  }\n  \n  /**\n   * Fetch fresh index from GitHub with conditional requests\n   */\n  private async fetchFresh(): Promise<CollectionIndex | null> {\n    try {\n      // Build headers for conditional request\n      const headers: Record<string, string> = {\n        'Accept': 'application/json',\n        'User-Agent': 'DollhouseMCP/1.0'\n      };\n      \n      // Add conditional headers if we have cache\n      if (this.cache?.etag) {\n        headers['If-None-Match'] = this.cache.etag;\n      }\n      if (this.cache?.lastModified) {\n        headers['If-Modified-Since'] = this.cache.lastModified;\n      }\n      \n      // Use fetch directly for better control over conditional requests\n      const response = await fetch(this.INDEX_URL, { headers });\n      \n      // 304 Not Modified - use cached data\n      if (response.status === 304) {\n        if (this.cache) {\n          // Update timestamp to extend cache validity\n          this.cache.fetchedAt = new Date();\n          await this.saveToDisk(this.cache);\n          logger.debug('Collection index not modified - refreshed cache timestamp');\n          return this.cache.data;\n        }\n      }\n      \n      if (!response.ok) {\n        throw new Error(`HTTP ${response.status}: ${response.statusText}`);\n      }\n      \n      const indexData = await response.json() as CollectionIndex;\n      \n      // Validate the index structure\n      if (!this.validateIndexStructure(indexData)) {\n        throw new Error('Invalid collection index structure received');\n      }\n      \n      // Create new cache entry\n      const newCache: CachedIndex = {\n        data: indexData,\n        fetchedAt: new Date(),\n        etag: response.headers.get('etag') || undefined,\n        lastModified: response.headers.get('last-modified') || undefined\n      };\n      \n      this.cache = newCache;\n      \n      // Save to persistent cache in background\n      this.saveToDisk(newCache).catch(error => {\n        logger.debug('Failed to save index cache to disk:', error);\n      });\n      \n      logger.debug(`Fresh collection index fetched with ${indexData.total_elements} elements`);\n      return indexData;\n      \n    } catch (error) {\n      logger.debug('Failed to fetch fresh collection index:', error);\n      return null;\n    }\n  }\n  \n  /**\n   * Validate the structure of a collection index\n   */\n  private validateIndexStructure(index: any): index is CollectionIndex {\n    return (\n      index &&\n      typeof index === 'object' &&\n      typeof index.version === 'string' &&\n      typeof index.generated === 'string' &&\n      typeof index.total_elements === 'number' &&\n      index.index &&\n      typeof index.index === 'object' &&\n      index.metadata &&\n      typeof index.metadata === 'object'\n    );\n  }\n  \n  /**\n   * Save cache to persistent storage\n   */\n  private async saveToDisk(cache: CachedIndex): Promise<void> {\n    try {\n      await this.ensureCacheDir();\n      \n      const cacheData = {\n        ...cache,\n        fetchedAt: cache.fetchedAt.toISOString() // Serialize date\n      };\n      \n      const data = JSON.stringify(cacheData, null, 2);\n      await fs.writeFile(this.cacheFile, data, 'utf8');\n      \n      logger.debug('Collection index cache saved to disk');\n    } catch (error) {\n      logger.debug('Failed to save collection index cache:', error);\n      // Don't throw - cache persistence failures shouldn't break functionality\n    }\n  }\n  \n  /**\n   * Load cache from persistent storage\n   */\n  private async loadFromDisk(): Promise<CachedIndex | null> {\n    try {\n      // Basic security check for path traversal\n      if (this.cacheFile.includes('..') || this.cacheFile.includes('\\0')) {\n        SecurityMonitor.logSecurityEvent({\n          type: 'PATH_TRAVERSAL_ATTEMPT',\n          severity: 'HIGH',\n          source: 'CollectionIndexCache.loadFromDisk',\n          details: `Potential path traversal attempt detected: ${this.cacheFile.substring(0, 100)}`\n        });\n        return null;\n      }\n      \n      const data = await fs.readFile(this.cacheFile, 'utf8');\n      const cacheData = JSON.parse(data);\n      \n      return {\n        ...cacheData,\n        fetchedAt: new Date(cacheData.fetchedAt) // Deserialize date\n      };\n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug('Failed to load collection index cache from disk:', error);\n      }\n      return null;\n    }\n  }\n  \n  /**\n   * Ensure cache directory exists\n   */\n  private async ensureCacheDir(): Promise<void> {\n    try {\n      await fs.mkdir(this.cacheDir, { recursive: true });\n    } catch (error) {\n      logger.error('Failed to create cache directory:', error);\n      throw error;\n    }\n  }\n  \n  /**\n   * Clear all cache data with performance monitoring\n   */\n  async clearCache(): Promise<void> {\n    const startTime = Date.now();\n    \n    this.cache = null;\n    this.memoryCache.clear();\n    \n    // Cancel any ongoing fetch\n    this.fetchPromise = null;\n    \n    try {\n      await fs.unlink(this.cacheFile);\n      logger.debug('Collection index cache cleared from disk');\n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug('Failed to clear collection index cache:', error);\n      }\n    }\n    \n    logger.debug('Collection cache cleared', {\n      duration: Date.now() - startTime,\n      memoryFreed: this.memoryCache.getMemoryUsageMB()\n    });\n  }\n  \n  /**\n   * Get comprehensive cache statistics for debugging and monitoring\n   */\n  getCacheStats(): { \n    isValid: boolean; \n    age: number; \n    hasCache: boolean; \n    elements: number;\n    memoryCache: any;\n    performanceMetrics: any;\n  } {\n    if (!this.cache) {\n      return { \n        isValid: false, \n        age: 0, \n        hasCache: false, \n        elements: 0,\n        memoryCache: this.memoryCache.getStats(),\n        performanceMetrics: null\n      };\n    }\n    \n    const age = Date.now() - this.cache.fetchedAt.getTime();\n    return {\n      isValid: this.isValid(),\n      age,\n      hasCache: true,\n      elements: this.cache.data.total_elements,\n      memoryCache: this.memoryCache.getStats(),\n      performanceMetrics: {\n        cacheHitRate: this.calculateCacheHitRate(),\n        averageAccessTime: this.calculateAverageAccessTime()\n      }\n    };\n  }\n  \n  // =====================================================\n  // PRIVATE HELPER METHODS FOR PERFORMANCE\n  // =====================================================\n  \n  /**\n   * Fetch fresh index with comprehensive fallback strategy\n   */\n  private async fetchFreshWithFallback(): Promise<CollectionIndex> {\n    try {\n      // Try to fetch fresh index\n      const freshIndex = await this.fetchFresh();\n      if (freshIndex) {\n        logger.debug('Collection index fetched successfully');\n        return freshIndex;\n      }\n    } catch (error) {\n      logger.warn('Fresh fetch failed, trying fallback', {\n        error: error instanceof Error ? error.message : String(error)\n      });\n    }\n    \n    // Fall back to stale cache if available\n    if (this.cache?.data) {\n      logger.debug('Using stale cached collection index as fallback');\n      return this.cache.data;\n    }\n    \n    throw new Error('No collection index available - fresh fetch failed and no cache exists');\n  }\n  \n  /**\n   * Record performance metrics for cache operations\n   */\n  private recordPerformanceMetrics(startTime: number, memoryBefore: number, operation: string): void {\n    const duration = Date.now() - startTime;\n    const memoryAfter = process.memoryUsage().heapUsed;\n    \n    logger.debug('Collection cache operation completed', {\n      operation,\n      duration,\n      memoryUsageMB: (memoryAfter - memoryBefore) / (1024 * 1024)\n    });\n    \n    // Record cache performance metrics\n    this.performanceMonitor.recordCachePerformance('collectionIndex', {\n      hitRate: operation.includes('hit') ? 1 : 0,\n      avgHitTime: operation.includes('hit') ? duration : 0,\n      avgMissTime: operation.includes('hit') ? 0 : duration,\n      totalHits: operation.includes('hit') ? 1 : 0,\n      totalMisses: operation.includes('hit') ? 0 : 1,\n      evictions: 0\n    });\n  }\n  \n  /**\n   * Calculate cache hit rate (placeholder for future implementation)\n   */\n  private calculateCacheHitRate(): number {\n    // This would be implemented with actual metrics tracking\n    return this.memoryCache.getStats().hitRate;\n  }\n  \n  /**\n   * Calculate average access time (placeholder for future implementation)\n   */\n  private calculateAverageAccessTime(): number {\n    // This would be implemented with actual timing metrics\n    return 5; // Placeholder value\n  }\n}"]}