@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
@@ -0,0 +1,499 @@
1
+ /**
2
+ * Collection Index Manager with Background Refresh and Robust Caching
3
+ *
4
+ * This manager implements:
5
+ * - 1-hour TTL with local file caching
6
+ * - Background refresh without blocking operations
7
+ * - Exponential backoff retry logic
8
+ * - Configurable timeouts via environment variables
9
+ * - Return stale cache while refreshing in background
10
+ * - Comprehensive error handling for production use
11
+ */
12
+ import * as fs from 'fs/promises';
13
+ import * as path from 'path';
14
+ import * as os from 'os';
15
+ import { logger } from '../utils/logger.js';
16
+ export class CollectionIndexManager {
17
+ INDEX_URL = 'https://raw.githubusercontent.com/DollhouseMCP/collection/main/public/collection-index.json';
18
+ TTL_MS;
19
+ FETCH_TIMEOUT_MS;
20
+ MAX_RETRIES;
21
+ BASE_RETRY_DELAY_MS;
22
+ MAX_RETRY_DELAY_MS;
23
+ CACHE_FILE;
24
+ cachedIndex = null;
25
+ backgroundRefreshPromise = null;
26
+ isRefreshing = false;
27
+ circuitBreakerFailures = 0;
28
+ circuitBreakerLastFailure = 0;
29
+ CIRCUIT_BREAKER_THRESHOLD = 5;
30
+ CIRCUIT_BREAKER_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes
31
+ REFRESH_THRESHOLD = 0.8; // Refresh when 80% of TTL has passed
32
+ JITTER_FACTOR = 0.25; // ±25% randomness for jitter
33
+ // Default configuration constants
34
+ DEFAULT_TTL_MS = 60 * 60 * 1000; // 1 hour
35
+ DEFAULT_MAX_RETRIES = 3;
36
+ DEFAULT_BASE_RETRY_DELAY_MS = 1000;
37
+ DEFAULT_MAX_RETRY_DELAY_MS = 30000;
38
+ DEFAULT_FETCH_TIMEOUT_MS = 5000; // 5 seconds
39
+ CHECKSUM_LENGTH = 8;
40
+ JSON_INDENT = 2;
41
+ constructor(config = {}) {
42
+ // Configuration with environment variable overrides
43
+ this.TTL_MS = config.ttlMs || this.DEFAULT_TTL_MS;
44
+ this.FETCH_TIMEOUT_MS = this.parseFetchTimeout(config.fetchTimeoutMs);
45
+ this.MAX_RETRIES = config.maxRetries || this.DEFAULT_MAX_RETRIES;
46
+ this.BASE_RETRY_DELAY_MS = config.baseRetryDelayMs || this.DEFAULT_BASE_RETRY_DELAY_MS;
47
+ this.MAX_RETRY_DELAY_MS = config.maxRetryDelayMs || this.DEFAULT_MAX_RETRY_DELAY_MS;
48
+ // Cache directory - use ~/.dollhouse/cache/collection-index.json as specified
49
+ const cacheDir = config.cacheDir || path.join(os.homedir(), '.dollhouse', 'cache');
50
+ this.CACHE_FILE = path.join(cacheDir, 'collection-index.json');
51
+ logger.debug('CollectionIndexManager initialized', {
52
+ ttlMs: this.TTL_MS,
53
+ fetchTimeoutMs: this.FETCH_TIMEOUT_MS,
54
+ cacheFile: this.CACHE_FILE,
55
+ maxRetries: this.MAX_RETRIES
56
+ });
57
+ }
58
+ /**
59
+ * Parse fetch timeout from config or environment variable
60
+ */
61
+ parseFetchTimeout(configValue) {
62
+ // Check environment variable first
63
+ const envTimeout = process.env.COLLECTION_FETCH_TIMEOUT;
64
+ if (envTimeout) {
65
+ const parsed = parseInt(envTimeout, 10);
66
+ if (!isNaN(parsed) && parsed > 0) {
67
+ logger.debug(`Using COLLECTION_FETCH_TIMEOUT from environment: ${parsed}ms`);
68
+ return parsed;
69
+ }
70
+ logger.warn(`Invalid COLLECTION_FETCH_TIMEOUT value: ${envTimeout}, using default`);
71
+ }
72
+ // Fall back to config value or default
73
+ return configValue || this.DEFAULT_FETCH_TIMEOUT_MS;
74
+ }
75
+ /**
76
+ * Get collection index with stale-while-revalidate pattern
77
+ * Returns cached data immediately if available, refreshes in background
78
+ */
79
+ async getIndex() {
80
+ try {
81
+ // Load from memory cache first
82
+ if (!this.cachedIndex) {
83
+ await this.loadFromDisk();
84
+ }
85
+ // Check if we should return stale cache while refreshing
86
+ const shouldRefresh = this.shouldRefreshCache();
87
+ if (this.cachedIndex && !this.isCacheExpired()) {
88
+ // Cache is valid, return immediately
89
+ logger.debug('Returning valid cached collection index');
90
+ // Start background refresh if needed but not already running
91
+ if (shouldRefresh && !this.isRefreshing) {
92
+ this.startBackgroundRefresh();
93
+ }
94
+ return this.cachedIndex.data;
95
+ }
96
+ // If we have stale cache, return it while refreshing in background
97
+ if (this.cachedIndex && this.isCacheExpired()) {
98
+ logger.debug('Returning stale cache while refreshing in background');
99
+ // Start background refresh if not already running
100
+ if (!this.isRefreshing) {
101
+ this.startBackgroundRefresh();
102
+ }
103
+ return this.cachedIndex.data;
104
+ }
105
+ // No cache available, must fetch synchronously
106
+ logger.debug('No cache available, fetching collection index synchronously');
107
+ const freshIndex = await this.fetchWithRetry();
108
+ await this.updateCache(freshIndex);
109
+ return freshIndex.data;
110
+ }
111
+ catch (error) {
112
+ logger.error('Failed to get collection index', { error: this.getErrorMessage(error) });
113
+ // If we have any cached data (even expired), return it as last resort
114
+ if (this.cachedIndex) {
115
+ logger.warn('Returning expired cache as last resort');
116
+ return this.cachedIndex.data;
117
+ }
118
+ throw new Error(`Collection index not available: ${this.getErrorMessage(error)}`);
119
+ }
120
+ }
121
+ /**
122
+ * Force refresh the collection index
123
+ */
124
+ async forceRefresh() {
125
+ logger.debug('Force refreshing collection index');
126
+ try {
127
+ const freshIndex = await this.fetchWithRetry();
128
+ await this.updateCache(freshIndex);
129
+ return freshIndex.data;
130
+ }
131
+ catch (error) {
132
+ logger.error('Force refresh failed', { error: this.getErrorMessage(error) });
133
+ // If we have cached data, return it
134
+ if (this.cachedIndex) {
135
+ logger.warn('Force refresh failed, returning cached data');
136
+ return this.cachedIndex.data;
137
+ }
138
+ throw error;
139
+ }
140
+ }
141
+ /**
142
+ * Check if cache should be refreshed (within TTL but getting close to expiry)
143
+ */
144
+ shouldRefreshCache() {
145
+ if (!this.cachedIndex)
146
+ return true;
147
+ const age = Date.now() - this.cachedIndex.timestamp;
148
+ const refreshThreshold = this.TTL_MS * this.REFRESH_THRESHOLD;
149
+ return age > refreshThreshold;
150
+ }
151
+ /**
152
+ * Check if cache is expired
153
+ */
154
+ isCacheExpired() {
155
+ if (!this.cachedIndex)
156
+ return true;
157
+ const age = Date.now() - this.cachedIndex.timestamp;
158
+ return age > this.TTL_MS;
159
+ }
160
+ /**
161
+ * Start background refresh without blocking
162
+ */
163
+ startBackgroundRefresh() {
164
+ if (this.isRefreshing) {
165
+ logger.debug('Background refresh already in progress');
166
+ return;
167
+ }
168
+ // Check circuit breaker
169
+ if (this.isCircuitBreakerOpen()) {
170
+ logger.debug('Circuit breaker open, skipping background refresh');
171
+ return;
172
+ }
173
+ this.isRefreshing = true;
174
+ this.backgroundRefreshPromise = this.performBackgroundRefresh()
175
+ .catch(error => {
176
+ logger.debug('Background refresh failed', { error: this.getErrorMessage(error) });
177
+ this.recordCircuitBreakerFailure();
178
+ })
179
+ .finally(() => {
180
+ this.isRefreshing = false;
181
+ this.backgroundRefreshPromise = null;
182
+ });
183
+ }
184
+ /**
185
+ * Perform background refresh
186
+ */
187
+ async performBackgroundRefresh() {
188
+ logger.debug('Starting background refresh of collection index');
189
+ try {
190
+ const freshIndex = await this.fetchWithRetry();
191
+ await this.updateCache(freshIndex);
192
+ // Reset circuit breaker on success
193
+ this.circuitBreakerFailures = 0;
194
+ logger.debug('Background refresh completed successfully');
195
+ }
196
+ catch (error) {
197
+ logger.debug('Background refresh failed', { error: this.getErrorMessage(error) });
198
+ throw error;
199
+ }
200
+ }
201
+ /**
202
+ * Fetch collection index with retry logic and exponential backoff
203
+ */
204
+ async fetchWithRetry() {
205
+ let lastError = null;
206
+ for (let attempt = 0; attempt <= this.MAX_RETRIES; attempt++) {
207
+ try {
208
+ if (attempt > 0) {
209
+ const delay = this.calculateRetryDelay(attempt);
210
+ logger.debug(`Retrying fetch in ${delay}ms (attempt ${attempt + 1}/${this.MAX_RETRIES + 1})`);
211
+ await this.sleep(delay);
212
+ }
213
+ return await this.fetchCollectionIndex();
214
+ }
215
+ catch (error) {
216
+ lastError = error instanceof Error ? error : new Error(String(error));
217
+ logger.debug(`Fetch attempt ${attempt + 1} failed`, {
218
+ error: this.getErrorMessage(lastError),
219
+ willRetry: attempt < this.MAX_RETRIES
220
+ });
221
+ }
222
+ }
223
+ throw lastError || new Error('All fetch attempts failed');
224
+ }
225
+ /**
226
+ * Calculate retry delay with exponential backoff and jitter
227
+ */
228
+ calculateRetryDelay(attempt) {
229
+ // Exponential backoff: baseDelay * (2 ^ attempt)
230
+ const exponentialDelay = this.BASE_RETRY_DELAY_MS * Math.pow(2, attempt - 1);
231
+ // Cap at maximum delay
232
+ const cappedDelay = Math.min(exponentialDelay, this.MAX_RETRY_DELAY_MS);
233
+ // Add jitter to prevent thundering herd
234
+ return this.addJitter(cappedDelay);
235
+ }
236
+ /**
237
+ * Add jitter (±25% randomness) to a delay to prevent thundering herd problems
238
+ */
239
+ addJitter(delay) {
240
+ const jitter = delay * this.JITTER_FACTOR * (Math.random() - 0.5);
241
+ return Math.max(0, delay + jitter);
242
+ }
243
+ /**
244
+ * Fetch collection index from GitHub
245
+ */
246
+ async fetchCollectionIndex() {
247
+ const controller = new AbortController();
248
+ const timeoutId = setTimeout(() => controller.abort(), this.FETCH_TIMEOUT_MS);
249
+ try {
250
+ // Build headers for conditional requests
251
+ const headers = {
252
+ 'Accept': 'application/json',
253
+ 'User-Agent': 'DollhouseMCP/1.0',
254
+ 'Cache-Control': 'no-cache'
255
+ };
256
+ // Add conditional headers if we have cached data
257
+ if (this.cachedIndex?.etag) {
258
+ headers['If-None-Match'] = this.cachedIndex.etag;
259
+ }
260
+ if (this.cachedIndex?.lastModified) {
261
+ headers['If-Modified-Since'] = this.cachedIndex.lastModified;
262
+ }
263
+ logger.debug('Fetching collection index', {
264
+ url: this.INDEX_URL,
265
+ timeout: this.FETCH_TIMEOUT_MS,
266
+ hasEtag: !!this.cachedIndex?.etag
267
+ });
268
+ const response = await fetch(this.INDEX_URL, {
269
+ headers,
270
+ signal: controller.signal
271
+ });
272
+ clearTimeout(timeoutId);
273
+ // Handle 304 Not Modified
274
+ if (response.status === 304 && this.cachedIndex) {
275
+ logger.debug('Collection index not modified (304), updating cache timestamp');
276
+ this.cachedIndex.timestamp = Date.now();
277
+ await this.saveToDisk();
278
+ return {
279
+ data: this.cachedIndex.data,
280
+ etag: this.cachedIndex.etag,
281
+ lastModified: this.cachedIndex.lastModified
282
+ };
283
+ }
284
+ if (!response.ok) {
285
+ throw new Error(`HTTP ${response.status}: ${response.statusText}`);
286
+ }
287
+ const indexData = await response.json();
288
+ // Validate the index structure
289
+ this.validateIndexStructure(indexData);
290
+ const etag = response.headers.get('etag') || undefined;
291
+ const lastModified = response.headers.get('last-modified') || undefined;
292
+ logger.debug('Collection index fetched successfully', {
293
+ totalElements: indexData.total_elements,
294
+ version: indexData.version,
295
+ hasEtag: !!etag
296
+ });
297
+ return { data: indexData, etag, lastModified };
298
+ }
299
+ catch (error) {
300
+ clearTimeout(timeoutId);
301
+ if (error instanceof Error && error.name === 'AbortError') {
302
+ throw new Error(`Fetch timeout after ${this.FETCH_TIMEOUT_MS}ms`);
303
+ }
304
+ throw error;
305
+ }
306
+ }
307
+ /**
308
+ * Validate collection index structure
309
+ */
310
+ validateIndexStructure(index) {
311
+ if (!index || typeof index !== 'object') {
312
+ throw new Error('Invalid index: not an object');
313
+ }
314
+ if (typeof index.version !== 'string') {
315
+ throw new Error('Invalid index: missing or invalid version');
316
+ }
317
+ if (typeof index.generated !== 'string') {
318
+ throw new Error('Invalid index: missing or invalid generated timestamp');
319
+ }
320
+ if (typeof index.total_elements !== 'number') {
321
+ throw new Error('Invalid index: missing or invalid total_elements');
322
+ }
323
+ if (!index.index || typeof index.index !== 'object') {
324
+ throw new Error('Invalid index: missing or invalid index object');
325
+ }
326
+ if (!index.metadata || typeof index.metadata !== 'object') {
327
+ throw new Error('Invalid index: missing or invalid metadata');
328
+ }
329
+ }
330
+ /**
331
+ * Update cache with new data
332
+ */
333
+ async updateCache(fetchResult) {
334
+ const checksum = this.calculateChecksum(fetchResult.data);
335
+ this.cachedIndex = {
336
+ data: fetchResult.data,
337
+ timestamp: Date.now(),
338
+ etag: fetchResult.etag,
339
+ lastModified: fetchResult.lastModified,
340
+ version: fetchResult.data.version,
341
+ checksum
342
+ };
343
+ await this.saveToDisk();
344
+ logger.debug('Collection index cache updated', {
345
+ version: fetchResult.data.version,
346
+ totalElements: fetchResult.data.total_elements,
347
+ checksum
348
+ });
349
+ }
350
+ /**
351
+ * Calculate checksum for data integrity verification
352
+ */
353
+ calculateChecksum(data) {
354
+ // Simple checksum based on key properties
355
+ const checksumData = {
356
+ version: data.version,
357
+ generated: data.generated,
358
+ total_elements: data.total_elements
359
+ };
360
+ return Buffer.from(JSON.stringify(checksumData)).toString('base64').substring(0, this.CHECKSUM_LENGTH);
361
+ }
362
+ /**
363
+ * Load cache from disk
364
+ */
365
+ async loadFromDisk() {
366
+ try {
367
+ const data = await fs.readFile(this.CACHE_FILE, 'utf8');
368
+ const cached = JSON.parse(data);
369
+ // Validate cache structure
370
+ if (!cached.data || !cached.timestamp || !cached.version) {
371
+ logger.debug('Invalid cache structure, ignoring');
372
+ return;
373
+ }
374
+ // Verify checksum if available
375
+ if (cached.checksum) {
376
+ const expectedChecksum = this.calculateChecksum(cached.data);
377
+ if (cached.checksum !== expectedChecksum) {
378
+ logger.debug('Cache checksum mismatch, ignoring cached data');
379
+ return;
380
+ }
381
+ }
382
+ this.cachedIndex = cached;
383
+ const age = Date.now() - cached.timestamp;
384
+ const isExpired = age > this.TTL_MS;
385
+ logger.debug('Loaded collection index from disk cache', {
386
+ version: cached.version,
387
+ age: Math.round(age / 1000),
388
+ isExpired,
389
+ totalElements: cached.data.total_elements
390
+ });
391
+ }
392
+ catch (error) {
393
+ if (error.code !== 'ENOENT') {
394
+ logger.debug('Failed to load cache from disk', { error: this.getErrorMessage(error) });
395
+ }
396
+ }
397
+ }
398
+ /**
399
+ * Save cache to disk
400
+ */
401
+ async saveToDisk() {
402
+ if (!this.cachedIndex)
403
+ return;
404
+ try {
405
+ // Ensure cache directory exists
406
+ await fs.mkdir(path.dirname(this.CACHE_FILE), { recursive: true });
407
+ const cacheData = JSON.stringify(this.cachedIndex, null, this.JSON_INDENT);
408
+ await fs.writeFile(this.CACHE_FILE, cacheData, 'utf8');
409
+ logger.debug('Collection index cache saved to disk');
410
+ }
411
+ catch (error) {
412
+ logger.debug('Failed to save cache to disk', { error: this.getErrorMessage(error) });
413
+ // Don't throw - cache persistence failures shouldn't break functionality
414
+ }
415
+ }
416
+ /**
417
+ * Circuit breaker logic
418
+ */
419
+ isCircuitBreakerOpen() {
420
+ if (this.circuitBreakerFailures < this.CIRCUIT_BREAKER_THRESHOLD) {
421
+ return false;
422
+ }
423
+ const timeSinceLastFailure = Date.now() - this.circuitBreakerLastFailure;
424
+ return timeSinceLastFailure < this.CIRCUIT_BREAKER_TIMEOUT_MS;
425
+ }
426
+ recordCircuitBreakerFailure() {
427
+ this.circuitBreakerFailures++;
428
+ this.circuitBreakerLastFailure = Date.now();
429
+ if (this.circuitBreakerFailures >= this.CIRCUIT_BREAKER_THRESHOLD) {
430
+ logger.warn('Circuit breaker opened due to repeated failures', {
431
+ failures: this.circuitBreakerFailures,
432
+ timeoutMs: this.CIRCUIT_BREAKER_TIMEOUT_MS
433
+ });
434
+ }
435
+ }
436
+ /**
437
+ * Utility methods
438
+ */
439
+ sleep(ms) {
440
+ return new Promise(resolve => setTimeout(resolve, ms));
441
+ }
442
+ getErrorMessage(error) {
443
+ if (error instanceof Error) {
444
+ return error.message;
445
+ }
446
+ return String(error);
447
+ }
448
+ /**
449
+ * Get cache statistics for monitoring
450
+ */
451
+ getCacheStats() {
452
+ if (!this.cachedIndex) {
453
+ return {
454
+ isValid: false,
455
+ age: 0,
456
+ hasCache: false,
457
+ isRefreshing: this.isRefreshing,
458
+ circuitBreakerFailures: this.circuitBreakerFailures,
459
+ circuitBreakerOpen: this.isCircuitBreakerOpen()
460
+ };
461
+ }
462
+ const age = Date.now() - this.cachedIndex.timestamp;
463
+ return {
464
+ isValid: !this.isCacheExpired(),
465
+ age: Math.round(age / 1000), // age in seconds
466
+ hasCache: true,
467
+ version: this.cachedIndex.version,
468
+ totalElements: this.cachedIndex.data.total_elements,
469
+ isRefreshing: this.isRefreshing,
470
+ circuitBreakerFailures: this.circuitBreakerFailures,
471
+ circuitBreakerOpen: this.isCircuitBreakerOpen()
472
+ };
473
+ }
474
+ /**
475
+ * Clear all cache data
476
+ */
477
+ async clearCache() {
478
+ this.cachedIndex = null;
479
+ this.circuitBreakerFailures = 0;
480
+ try {
481
+ await fs.unlink(this.CACHE_FILE);
482
+ logger.debug('Collection index cache file deleted');
483
+ }
484
+ catch (error) {
485
+ if (error.code !== 'ENOENT') {
486
+ logger.debug('Failed to delete cache file', { error: this.getErrorMessage(error) });
487
+ }
488
+ }
489
+ }
490
+ /**
491
+ * Wait for any ongoing background refresh to complete
492
+ */
493
+ async waitForBackgroundRefresh() {
494
+ if (this.backgroundRefreshPromise) {
495
+ await this.backgroundRefreshPromise;
496
+ }
497
+ }
498
+ }
499
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CollectionIndexManager.js","sourceRoot":"","sources":["../../src/collection/CollectionIndexManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Collection Index Manager with Background Refresh and Robust Caching\n * \n * This manager implements:\n * - 1-hour TTL with local file caching\n * - Background refresh without blocking operations\n * - Exponential backoff retry logic\n * - Configurable timeouts via environment variables\n * - Return stale cache while refreshing in background\n * - Comprehensive error handling for production use\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport * as os from 'os';\nimport { CollectionIndex } from '../types/collection';\nimport { logger } from '../utils/logger.js';\n\nexport interface CollectionIndexCacheEntry {\n  data: CollectionIndex;\n  timestamp: number;\n  etag?: string;\n  lastModified?: string;\n  version: string;\n  checksum: string;\n}\n\nexport interface CollectionIndexManagerConfig {\n  ttlMs?: number;\n  fetchTimeoutMs?: number;\n  maxRetries?: number;\n  baseRetryDelayMs?: number;\n  maxRetryDelayMs?: number;\n  cacheDir?: string;\n}\n\nexport class CollectionIndexManager {\n  private readonly INDEX_URL = 'https://raw.githubusercontent.com/DollhouseMCP/collection/main/public/collection-index.json';\n  private readonly TTL_MS: number;\n  private readonly FETCH_TIMEOUT_MS: number;\n  private readonly MAX_RETRIES: number;\n  private readonly BASE_RETRY_DELAY_MS: number;\n  private readonly MAX_RETRY_DELAY_MS: number;\n  private readonly CACHE_FILE: string;\n  \n  private cachedIndex: CollectionIndexCacheEntry | null = null;\n  private backgroundRefreshPromise: Promise<void> | null = null;\n  private isRefreshing = false;\n  private circuitBreakerFailures = 0;\n  private circuitBreakerLastFailure = 0;\n  private readonly CIRCUIT_BREAKER_THRESHOLD = 5;\n  private readonly CIRCUIT_BREAKER_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes\n  private readonly REFRESH_THRESHOLD = 0.8; // Refresh when 80% of TTL has passed\n  private readonly JITTER_FACTOR = 0.25; // ±25% randomness for jitter\n  \n  // Default configuration constants\n  private readonly DEFAULT_TTL_MS = 60 * 60 * 1000; // 1 hour\n  private readonly DEFAULT_MAX_RETRIES = 3;\n  private readonly DEFAULT_BASE_RETRY_DELAY_MS = 1000;\n  private readonly DEFAULT_MAX_RETRY_DELAY_MS = 30000;\n  private readonly DEFAULT_FETCH_TIMEOUT_MS = 5000; // 5 seconds\n  private readonly CHECKSUM_LENGTH = 8;\n  private readonly JSON_INDENT = 2;\n  \n  constructor(config: CollectionIndexManagerConfig = {}) {\n    // Configuration with environment variable overrides\n    this.TTL_MS = config.ttlMs || this.DEFAULT_TTL_MS;\n    this.FETCH_TIMEOUT_MS = this.parseFetchTimeout(config.fetchTimeoutMs);\n    this.MAX_RETRIES = config.maxRetries || this.DEFAULT_MAX_RETRIES;\n    this.BASE_RETRY_DELAY_MS = config.baseRetryDelayMs || this.DEFAULT_BASE_RETRY_DELAY_MS;\n    this.MAX_RETRY_DELAY_MS = config.maxRetryDelayMs || this.DEFAULT_MAX_RETRY_DELAY_MS;\n    \n    // Cache directory - use ~/.dollhouse/cache/collection-index.json as specified\n    const cacheDir = config.cacheDir || path.join(os.homedir(), '.dollhouse', 'cache');\n    this.CACHE_FILE = path.join(cacheDir, 'collection-index.json');\n    \n    logger.debug('CollectionIndexManager initialized', {\n      ttlMs: this.TTL_MS,\n      fetchTimeoutMs: this.FETCH_TIMEOUT_MS,\n      cacheFile: this.CACHE_FILE,\n      maxRetries: this.MAX_RETRIES\n    });\n  }\n  \n  /**\n   * Parse fetch timeout from config or environment variable\n   */\n  private parseFetchTimeout(configValue?: number): number {\n    // Check environment variable first\n    const envTimeout = process.env.COLLECTION_FETCH_TIMEOUT;\n    if (envTimeout) {\n      const parsed = parseInt(envTimeout, 10);\n      if (!isNaN(parsed) && parsed > 0) {\n        logger.debug(`Using COLLECTION_FETCH_TIMEOUT from environment: ${parsed}ms`);\n        return parsed;\n      }\n      logger.warn(`Invalid COLLECTION_FETCH_TIMEOUT value: ${envTimeout}, using default`);\n    }\n    \n    // Fall back to config value or default\n    return configValue || this.DEFAULT_FETCH_TIMEOUT_MS;\n  }\n  \n  /**\n   * Get collection index with stale-while-revalidate pattern\n   * Returns cached data immediately if available, refreshes in background\n   */\n  async getIndex(): Promise<CollectionIndex> {\n    try {\n      // Load from memory cache first\n      if (!this.cachedIndex) {\n        await this.loadFromDisk();\n      }\n      \n      // Check if we should return stale cache while refreshing\n      const shouldRefresh = this.shouldRefreshCache();\n      \n      if (this.cachedIndex && !this.isCacheExpired()) {\n        // Cache is valid, return immediately\n        logger.debug('Returning valid cached collection index');\n        \n        // Start background refresh if needed but not already running\n        if (shouldRefresh && !this.isRefreshing) {\n          this.startBackgroundRefresh();\n        }\n        \n        return this.cachedIndex.data;\n      }\n      \n      // If we have stale cache, return it while refreshing in background\n      if (this.cachedIndex && this.isCacheExpired()) {\n        logger.debug('Returning stale cache while refreshing in background');\n        \n        // Start background refresh if not already running\n        if (!this.isRefreshing) {\n          this.startBackgroundRefresh();\n        }\n        \n        return this.cachedIndex.data;\n      }\n      \n      // No cache available, must fetch synchronously\n      logger.debug('No cache available, fetching collection index synchronously');\n      const freshIndex = await this.fetchWithRetry();\n      await this.updateCache(freshIndex);\n      return freshIndex.data;\n      \n    } catch (error) {\n      logger.error('Failed to get collection index', { error: this.getErrorMessage(error) });\n      \n      // If we have any cached data (even expired), return it as last resort\n      if (this.cachedIndex) {\n        logger.warn('Returning expired cache as last resort');\n        return this.cachedIndex.data;\n      }\n      \n      throw new Error(`Collection index not available: ${this.getErrorMessage(error)}`);\n    }\n  }\n  \n  /**\n   * Force refresh the collection index\n   */\n  async forceRefresh(): Promise<CollectionIndex> {\n    logger.debug('Force refreshing collection index');\n    \n    try {\n      const freshIndex = await this.fetchWithRetry();\n      await this.updateCache(freshIndex);\n      return freshIndex.data;\n    } catch (error) {\n      logger.error('Force refresh failed', { error: this.getErrorMessage(error) });\n      \n      // If we have cached data, return it\n      if (this.cachedIndex) {\n        logger.warn('Force refresh failed, returning cached data');\n        return this.cachedIndex.data;\n      }\n      \n      throw error;\n    }\n  }\n  \n  /**\n   * Check if cache should be refreshed (within TTL but getting close to expiry)\n   */\n  private shouldRefreshCache(): boolean {\n    if (!this.cachedIndex) return true;\n    \n    const age = Date.now() - this.cachedIndex.timestamp;\n    const refreshThreshold = this.TTL_MS * this.REFRESH_THRESHOLD;\n    \n    return age > refreshThreshold;\n  }\n  \n  /**\n   * Check if cache is expired\n   */\n  private isCacheExpired(): boolean {\n    if (!this.cachedIndex) return true;\n    \n    const age = Date.now() - this.cachedIndex.timestamp;\n    return age > this.TTL_MS;\n  }\n  \n  /**\n   * Start background refresh without blocking\n   */\n  private startBackgroundRefresh(): void {\n    if (this.isRefreshing) {\n      logger.debug('Background refresh already in progress');\n      return;\n    }\n    \n    // Check circuit breaker\n    if (this.isCircuitBreakerOpen()) {\n      logger.debug('Circuit breaker open, skipping background refresh');\n      return;\n    }\n    \n    this.isRefreshing = true;\n    \n    this.backgroundRefreshPromise = this.performBackgroundRefresh()\n      .catch(error => {\n        logger.debug('Background refresh failed', { error: this.getErrorMessage(error) });\n        this.recordCircuitBreakerFailure();\n      })\n      .finally(() => {\n        this.isRefreshing = false;\n        this.backgroundRefreshPromise = null;\n      });\n  }\n  \n  /**\n   * Perform background refresh\n   */\n  private async performBackgroundRefresh(): Promise<void> {\n    logger.debug('Starting background refresh of collection index');\n    \n    try {\n      const freshIndex = await this.fetchWithRetry();\n      await this.updateCache(freshIndex);\n      \n      // Reset circuit breaker on success\n      this.circuitBreakerFailures = 0;\n      \n      logger.debug('Background refresh completed successfully');\n    } catch (error) {\n      logger.debug('Background refresh failed', { error: this.getErrorMessage(error) });\n      throw error;\n    }\n  }\n  \n  /**\n   * Fetch collection index with retry logic and exponential backoff\n   */\n  private async fetchWithRetry(): Promise<{ data: CollectionIndex; etag?: string; lastModified?: string }> {\n    let lastError: Error | null = null;\n    \n    for (let attempt = 0; attempt <= this.MAX_RETRIES; attempt++) {\n      try {\n        if (attempt > 0) {\n          const delay = this.calculateRetryDelay(attempt);\n          logger.debug(`Retrying fetch in ${delay}ms (attempt ${attempt + 1}/${this.MAX_RETRIES + 1})`);\n          await this.sleep(delay);\n        }\n        \n        return await this.fetchCollectionIndex();\n      } catch (error) {\n        lastError = error instanceof Error ? error : new Error(String(error));\n        logger.debug(`Fetch attempt ${attempt + 1} failed`, { \n          error: this.getErrorMessage(lastError),\n          willRetry: attempt < this.MAX_RETRIES\n        });\n      }\n    }\n    \n    throw lastError || new Error('All fetch attempts failed');\n  }\n  \n  /**\n   * Calculate retry delay with exponential backoff and jitter\n   */\n  private calculateRetryDelay(attempt: number): number {\n    // Exponential backoff: baseDelay * (2 ^ attempt)\n    const exponentialDelay = this.BASE_RETRY_DELAY_MS * Math.pow(2, attempt - 1);\n    \n    // Cap at maximum delay\n    const cappedDelay = Math.min(exponentialDelay, this.MAX_RETRY_DELAY_MS);\n    \n    // Add jitter to prevent thundering herd\n    return this.addJitter(cappedDelay);\n  }\n  \n  /**\n   * Add jitter (±25% randomness) to a delay to prevent thundering herd problems\n   */\n  private addJitter(delay: number): number {\n    const jitter = delay * this.JITTER_FACTOR * (Math.random() - 0.5);\n    return Math.max(0, delay + jitter);\n  }\n  \n  /**\n   * Fetch collection index from GitHub\n   */\n  private async fetchCollectionIndex(): Promise<{ data: CollectionIndex; etag?: string; lastModified?: string }> {\n    const controller = new AbortController();\n    const timeoutId = setTimeout(() => controller.abort(), this.FETCH_TIMEOUT_MS);\n    \n    try {\n      // Build headers for conditional requests\n      const headers: Record<string, string> = {\n        'Accept': 'application/json',\n        'User-Agent': 'DollhouseMCP/1.0',\n        'Cache-Control': 'no-cache'\n      };\n      \n      // Add conditional headers if we have cached data\n      if (this.cachedIndex?.etag) {\n        headers['If-None-Match'] = this.cachedIndex.etag;\n      }\n      if (this.cachedIndex?.lastModified) {\n        headers['If-Modified-Since'] = this.cachedIndex.lastModified;\n      }\n      \n      logger.debug('Fetching collection index', { \n        url: this.INDEX_URL,\n        timeout: this.FETCH_TIMEOUT_MS,\n        hasEtag: !!this.cachedIndex?.etag\n      });\n      \n      const response = await fetch(this.INDEX_URL, {\n        headers,\n        signal: controller.signal\n      });\n      \n      clearTimeout(timeoutId);\n      \n      // Handle 304 Not Modified\n      if (response.status === 304 && this.cachedIndex) {\n        logger.debug('Collection index not modified (304), updating cache timestamp');\n        this.cachedIndex.timestamp = Date.now();\n        await this.saveToDisk();\n        return {\n          data: this.cachedIndex.data,\n          etag: this.cachedIndex.etag,\n          lastModified: this.cachedIndex.lastModified\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      this.validateIndexStructure(indexData);\n      \n      const etag = response.headers.get('etag') || undefined;\n      const lastModified = response.headers.get('last-modified') || undefined;\n      \n      logger.debug('Collection index fetched successfully', {\n        totalElements: indexData.total_elements,\n        version: indexData.version,\n        hasEtag: !!etag\n      });\n      \n      return { data: indexData, etag, lastModified };\n      \n    } catch (error) {\n      clearTimeout(timeoutId);\n      \n      if (error instanceof Error && error.name === 'AbortError') {\n        throw new Error(`Fetch timeout after ${this.FETCH_TIMEOUT_MS}ms`);\n      }\n      \n      throw error;\n    }\n  }\n  \n  /**\n   * Validate collection index structure\n   */\n  private validateIndexStructure(index: any): asserts index is CollectionIndex {\n    if (!index || typeof index !== 'object') {\n      throw new Error('Invalid index: not an object');\n    }\n    \n    if (typeof index.version !== 'string') {\n      throw new Error('Invalid index: missing or invalid version');\n    }\n    \n    if (typeof index.generated !== 'string') {\n      throw new Error('Invalid index: missing or invalid generated timestamp');\n    }\n    \n    if (typeof index.total_elements !== 'number') {\n      throw new Error('Invalid index: missing or invalid total_elements');\n    }\n    \n    if (!index.index || typeof index.index !== 'object') {\n      throw new Error('Invalid index: missing or invalid index object');\n    }\n    \n    if (!index.metadata || typeof index.metadata !== 'object') {\n      throw new Error('Invalid index: missing or invalid metadata');\n    }\n  }\n  \n  /**\n   * Update cache with new data\n   */\n  private async updateCache(fetchResult: { data: CollectionIndex; etag?: string; lastModified?: string }): Promise<void> {\n    const checksum = this.calculateChecksum(fetchResult.data);\n    \n    this.cachedIndex = {\n      data: fetchResult.data,\n      timestamp: Date.now(),\n      etag: fetchResult.etag,\n      lastModified: fetchResult.lastModified,\n      version: fetchResult.data.version,\n      checksum\n    };\n    \n    await this.saveToDisk();\n    \n    logger.debug('Collection index cache updated', {\n      version: fetchResult.data.version,\n      totalElements: fetchResult.data.total_elements,\n      checksum\n    });\n  }\n  \n  /**\n   * Calculate checksum for data integrity verification\n   */\n  private calculateChecksum(data: CollectionIndex): string {\n    // Simple checksum based on key properties\n    const checksumData = {\n      version: data.version,\n      generated: data.generated,\n      total_elements: data.total_elements\n    };\n    return Buffer.from(JSON.stringify(checksumData)).toString('base64').substring(0, this.CHECKSUM_LENGTH);\n  }\n  \n  /**\n   * Load cache from disk\n   */\n  private async loadFromDisk(): Promise<void> {\n    try {\n      const data = await fs.readFile(this.CACHE_FILE, 'utf8');\n      const cached = JSON.parse(data) as CollectionIndexCacheEntry;\n      \n      // Validate cache structure\n      if (!cached.data || !cached.timestamp || !cached.version) {\n        logger.debug('Invalid cache structure, ignoring');\n        return;\n      }\n      \n      // Verify checksum if available\n      if (cached.checksum) {\n        const expectedChecksum = this.calculateChecksum(cached.data);\n        if (cached.checksum !== expectedChecksum) {\n          logger.debug('Cache checksum mismatch, ignoring cached data');\n          return;\n        }\n      }\n      \n      this.cachedIndex = cached;\n      \n      const age = Date.now() - cached.timestamp;\n      const isExpired = age > this.TTL_MS;\n      \n      logger.debug('Loaded collection index from disk cache', {\n        version: cached.version,\n        age: Math.round(age / 1000),\n        isExpired,\n        totalElements: cached.data.total_elements\n      });\n      \n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug('Failed to load cache from disk', { error: this.getErrorMessage(error) });\n      }\n    }\n  }\n  \n  /**\n   * Save cache to disk\n   */\n  private async saveToDisk(): Promise<void> {\n    if (!this.cachedIndex) return;\n    \n    try {\n      // Ensure cache directory exists\n      await fs.mkdir(path.dirname(this.CACHE_FILE), { recursive: true });\n      \n      const cacheData = JSON.stringify(this.cachedIndex, null, this.JSON_INDENT);\n      await fs.writeFile(this.CACHE_FILE, cacheData, 'utf8');\n      \n      logger.debug('Collection index cache saved to disk');\n    } catch (error) {\n      logger.debug('Failed to save cache to disk', { error: this.getErrorMessage(error) });\n      // Don't throw - cache persistence failures shouldn't break functionality\n    }\n  }\n  \n  /**\n   * Circuit breaker logic\n   */\n  private isCircuitBreakerOpen(): boolean {\n    if (this.circuitBreakerFailures < this.CIRCUIT_BREAKER_THRESHOLD) {\n      return false;\n    }\n    \n    const timeSinceLastFailure = Date.now() - this.circuitBreakerLastFailure;\n    return timeSinceLastFailure < this.CIRCUIT_BREAKER_TIMEOUT_MS;\n  }\n  \n  private recordCircuitBreakerFailure(): void {\n    this.circuitBreakerFailures++;\n    this.circuitBreakerLastFailure = Date.now();\n    \n    if (this.circuitBreakerFailures >= this.CIRCUIT_BREAKER_THRESHOLD) {\n      logger.warn('Circuit breaker opened due to repeated failures', {\n        failures: this.circuitBreakerFailures,\n        timeoutMs: this.CIRCUIT_BREAKER_TIMEOUT_MS\n      });\n    }\n  }\n  \n  /**\n   * Utility methods\n   */\n  private sleep(ms: number): Promise<void> {\n    return new Promise(resolve => setTimeout(resolve, ms));\n  }\n  \n  private getErrorMessage(error: unknown): string {\n    if (error instanceof Error) {\n      return error.message;\n    }\n    return String(error);\n  }\n  \n  /**\n   * Get cache statistics for monitoring\n   */\n  getCacheStats(): {\n    isValid: boolean;\n    age: number;\n    hasCache: boolean;\n    version?: string;\n    totalElements?: number;\n    isRefreshing: boolean;\n    circuitBreakerFailures: number;\n    circuitBreakerOpen: boolean;\n  } {\n    if (!this.cachedIndex) {\n      return {\n        isValid: false,\n        age: 0,\n        hasCache: false,\n        isRefreshing: this.isRefreshing,\n        circuitBreakerFailures: this.circuitBreakerFailures,\n        circuitBreakerOpen: this.isCircuitBreakerOpen()\n      };\n    }\n    \n    const age = Date.now() - this.cachedIndex.timestamp;\n    \n    return {\n      isValid: !this.isCacheExpired(),\n      age: Math.round(age / 1000), // age in seconds\n      hasCache: true,\n      version: this.cachedIndex.version,\n      totalElements: this.cachedIndex.data.total_elements,\n      isRefreshing: this.isRefreshing,\n      circuitBreakerFailures: this.circuitBreakerFailures,\n      circuitBreakerOpen: this.isCircuitBreakerOpen()\n    };\n  }\n  \n  /**\n   * Clear all cache data\n   */\n  async clearCache(): Promise<void> {\n    this.cachedIndex = null;\n    this.circuitBreakerFailures = 0;\n    \n    try {\n      await fs.unlink(this.CACHE_FILE);\n      logger.debug('Collection index cache file deleted');\n    } catch (error) {\n      if ((error as any).code !== 'ENOENT') {\n        logger.debug('Failed to delete cache file', { error: this.getErrorMessage(error) });\n      }\n    }\n  }\n  \n  /**\n   * Wait for any ongoing background refresh to complete\n   */\n  async waitForBackgroundRefresh(): Promise<void> {\n    if (this.backgroundRefreshPromise) {\n      await this.backgroundRefreshPromise;\n    }\n  }\n}"]}
@@ -3,11 +3,18 @@
3
3
  */
4
4
  import { GitHubClient } from './GitHubClient.js';
5
5
  import { CollectionCache } from '../cache/CollectionCache.js';
6
+ import { SearchResults, SearchOptions } from '../types/collection.js';
6
7
  export declare class CollectionSearch {
7
8
  private githubClient;
8
9
  private collectionCache;
10
+ private indexCache;
9
11
  private searchBaseUrl;
10
12
  constructor(githubClient: GitHubClient, collectionCache?: CollectionCache);
13
+ /**
14
+ * Enhanced search using collection index with pagination and filtering
15
+ * Falls back to API search and cache when index is unavailable
16
+ */
17
+ searchCollectionWithOptions(query: string, options?: SearchOptions): Promise<SearchResults>;
11
18
  /**
12
19
  * Search collection for content matching query
13
20
  * Falls back to cached data when GitHub API is not available or not authenticated
@@ -21,6 +28,10 @@ export declare class CollectionSearch {
21
28
  * Search seed data for matching items with fuzzy matching
22
29
  */
23
30
  private searchSeedData;
31
+ /**
32
+ * Fuzzy matching algorithm for partial string matches
33
+ */
34
+ private fuzzyMatch;
24
35
  /**
25
36
  * Convert cache items to GitHub API format for consistent response structure
26
37
  */
@@ -33,5 +44,49 @@ export declare class CollectionSearch {
33
44
  * Format search results
34
45
  */
35
46
  formatSearchResults(items: any[], query: string, personaIndicator?: string): string;
47
+ /**
48
+ * Search from collection index with full featured search and pagination
49
+ */
50
+ private searchFromIndex;
51
+ /**
52
+ * Flatten index entries from all categories into a single array
53
+ */
54
+ private flattenIndexEntries;
55
+ /**
56
+ * Perform search matching on index entries
57
+ */
58
+ private performIndexSearch;
59
+ /**
60
+ * Sort search results by relevance, name, or date
61
+ */
62
+ private sortSearchResults;
63
+ /**
64
+ * Calculate relevance score for search results
65
+ */
66
+ private calculateRelevanceScore;
67
+ /**
68
+ * Convert legacy search results to new SearchResults format
69
+ */
70
+ private convertLegacyResults;
71
+ /**
72
+ * Extract element type from file path
73
+ */
74
+ private extractTypeFromPath;
75
+ /**
76
+ * Extract category from file path
77
+ */
78
+ private extractCategoryFromPath;
79
+ /**
80
+ * Create empty search results for error cases
81
+ */
82
+ private createEmptySearchResults;
83
+ /**
84
+ * Enhanced format for search results with pagination info
85
+ */
86
+ formatSearchResultsWithPagination(results: SearchResults, personaIndicator?: string): string;
87
+ /**
88
+ * Get cache statistics for debugging
89
+ */
90
+ getCacheStats(): Promise<any>;
36
91
  }
37
92
  //# sourceMappingURL=CollectionSearch.d.ts.map