@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,1434 @@
1
+ /**
2
+ * Unified Index Manager - Combines local, GitHub, and collection portfolio indexing
3
+ *
4
+ * Features:
5
+ * - Unified search across local, GitHub, and collection portfolios
6
+ * - Intelligent result merging and deduplication
7
+ * - Version conflict detection and resolution
8
+ * - Performance optimization with parallel indexing
9
+ * - Advanced fallback strategies for resilient operation
10
+ * - Comprehensive search capabilities with pagination
11
+ * - Smart result ranking and duplicate detection
12
+ */
13
+ import { PortfolioIndexManager } from './PortfolioIndexManager.js';
14
+ import { GitHubPortfolioIndexer } from './GitHubPortfolioIndexer.js';
15
+ import { CollectionIndexCache } from '../cache/CollectionIndexCache.js';
16
+ import { GitHubClient } from '../collection/GitHubClient.js';
17
+ import { APICache } from '../cache/APICache.js';
18
+ import { ElementType } from './types.js';
19
+ import { logger } from '../utils/logger.js';
20
+ import { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';
21
+ import { CacheFactory } from '../cache/LRUCache.js';
22
+ import { PerformanceMonitor } from '../utils/PerformanceMonitor.js';
23
+ import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
24
+ import { SecurityMonitor } from '../security/securityMonitor.js';
25
+ export class UnifiedIndexManager {
26
+ static instance = null;
27
+ localIndexManager;
28
+ githubIndexer;
29
+ collectionIndexCache;
30
+ githubClient;
31
+ // Performance monitoring and caching
32
+ performanceMonitor;
33
+ resultCache;
34
+ indexCache;
35
+ BATCH_SIZE = 50; // For streaming results
36
+ MAX_CONCURRENT_SOURCES = 3;
37
+ constructor() {
38
+ this.localIndexManager = PortfolioIndexManager.getInstance();
39
+ this.githubIndexer = GitHubPortfolioIndexer.getInstance();
40
+ // Initialize GitHubClient with required dependencies
41
+ const apiCache = new APICache();
42
+ const rateLimitTracker = new Map();
43
+ this.githubClient = new GitHubClient(apiCache, rateLimitTracker);
44
+ this.collectionIndexCache = new CollectionIndexCache(this.githubClient);
45
+ // Initialize performance monitoring and caching
46
+ this.performanceMonitor = PerformanceMonitor.getInstance();
47
+ this.performanceMonitor.startMonitoring();
48
+ this.resultCache = CacheFactory.createSearchResultCache({
49
+ maxSize: 200,
50
+ maxMemoryMB: 15,
51
+ ttlMs: 5 * 60 * 1000, // 5 minutes
52
+ onEviction: (key, value) => {
53
+ logger.debug('Search result cache eviction', { key, resultCount: value.length });
54
+ }
55
+ });
56
+ this.indexCache = CacheFactory.createIndexCache({
57
+ maxSize: 100,
58
+ maxMemoryMB: 20,
59
+ ttlMs: 15 * 60 * 1000, // 15 minutes
60
+ onEviction: (key, value) => {
61
+ logger.debug('Index cache eviction', { key });
62
+ }
63
+ });
64
+ logger.debug('UnifiedIndexManager created with performance optimization');
65
+ }
66
+ static getInstance() {
67
+ if (!this.instance) {
68
+ this.instance = new UnifiedIndexManager();
69
+ }
70
+ return this.instance;
71
+ }
72
+ /**
73
+ * Enhanced search across local, GitHub, and collection portfolios with performance optimization
74
+ */
75
+ async search(searchOptions) {
76
+ const startTime = Date.now();
77
+ const memoryBefore = process.memoryUsage().heapUsed;
78
+ const { query, includeLocal = true, includeGitHub = true, includeCollection = false } = searchOptions;
79
+ // Normalize query to prevent Unicode-based attacks
80
+ const validationResult = UnicodeValidator.normalize(query);
81
+ const normalizedQuery = validationResult.normalizedContent;
82
+ // Use normalized query in all subsequent operations
83
+ const normalizedSearchOptions = {
84
+ ...searchOptions,
85
+ query: normalizedQuery
86
+ };
87
+ // SECURITY FIX (DMCP-SEC-006): Add audit logging for security monitoring
88
+ // Log unified search operations for security audit trail
89
+ SecurityMonitor.logSecurityEvent({
90
+ type: 'PORTFOLIO_FETCH_SUCCESS',
91
+ severity: 'LOW',
92
+ source: 'UnifiedIndexManager.search',
93
+ details: `Unified search performed with query length: ${normalizedQuery.length}, sources: ${JSON.stringify({
94
+ local: includeLocal,
95
+ github: includeGitHub,
96
+ collection: includeCollection
97
+ })}`
98
+ });
99
+ logger.debug('Starting optimized unified portfolio search', normalizedSearchOptions);
100
+ // Check cache first (use normalized search options)
101
+ const cacheKey = this.createCacheKey(normalizedSearchOptions);
102
+ const cached = this.resultCache.get(cacheKey);
103
+ if (cached) {
104
+ const duration = Date.now() - startTime;
105
+ this.recordSearchMetrics({
106
+ query: normalizedQuery,
107
+ duration,
108
+ resultCount: cached.length,
109
+ sources: this.getEnabledSources(normalizedSearchOptions),
110
+ cacheHit: true,
111
+ memoryBefore,
112
+ memoryAfter: process.memoryUsage().heapUsed,
113
+ timestamp: new Date()
114
+ });
115
+ logger.debug('Using cached search results', { resultCount: cached.length });
116
+ return cached;
117
+ }
118
+ try {
119
+ // Use streaming search for better performance with large result sets
120
+ if (normalizedSearchOptions.streamResults) {
121
+ return await this.streamSearch(normalizedSearchOptions);
122
+ }
123
+ // Lazy loading: Only load indices when needed
124
+ const searchPromises = [];
125
+ const enabledSources = this.getEnabledSources(normalizedSearchOptions);
126
+ // Limit concurrent source searches for memory efficiency
127
+ const concurrentLimit = Math.min(this.MAX_CONCURRENT_SOURCES, enabledSources.length);
128
+ const sourceBatches = this.batchSources(enabledSources, concurrentLimit);
129
+ const allResults = [];
130
+ const sourceCount = { local: 0, github: 0, collection: 0 };
131
+ // Process sources in batches to control memory usage
132
+ for (const batch of sourceBatches) {
133
+ const batchPromises = batch.map(source => this.searchWithFallback(source, normalizedQuery, normalizedSearchOptions));
134
+ const batchResults = await Promise.allSettled(batchPromises);
135
+ batchResults.forEach((result, index) => {
136
+ const sourceName = batch[index];
137
+ if (result.status === 'fulfilled') {
138
+ sourceCount[sourceName] += result.value.length;
139
+ allResults.push(...result.value);
140
+ }
141
+ else {
142
+ logger.warn(`Search failed for source ${sourceName}`, {
143
+ error: result.reason instanceof Error ? result.reason.message : String(result.reason)
144
+ });
145
+ }
146
+ });
147
+ // Memory check between batches
148
+ const currentMemory = process.memoryUsage().heapUsed / (1024 * 1024);
149
+ if (currentMemory > 200) { // 200MB threshold
150
+ logger.warn('High memory usage during search, triggering cleanup', {
151
+ memoryMB: currentMemory
152
+ });
153
+ this.triggerMemoryCleanup();
154
+ }
155
+ }
156
+ // Apply advanced processing with memory-efficient batching
157
+ const processedResults = await this.processSearchResultsOptimized(allResults, normalizedSearchOptions);
158
+ // Apply pagination
159
+ const paginatedResults = this.applyPagination(processedResults, normalizedSearchOptions);
160
+ // Cache results with memory limit check
161
+ if (paginatedResults.length < 1000) { // Don't cache very large result sets
162
+ this.resultCache.set(cacheKey, paginatedResults);
163
+ }
164
+ const duration = Date.now() - startTime;
165
+ const memoryAfter = process.memoryUsage().heapUsed;
166
+ this.recordSearchMetrics({
167
+ query: normalizedQuery,
168
+ duration,
169
+ resultCount: paginatedResults.length,
170
+ sources: enabledSources,
171
+ cacheHit: false,
172
+ memoryBefore,
173
+ memoryAfter,
174
+ timestamp: new Date()
175
+ });
176
+ logger.info('Optimized unified portfolio search completed', {
177
+ query: normalizedQuery.substring(0, 50),
178
+ sources: { ...sourceCount, total: allResults.length },
179
+ finalResults: paginatedResults.length,
180
+ duration: `${duration}ms`,
181
+ memoryUsageMB: (memoryAfter - memoryBefore) / (1024 * 1024)
182
+ });
183
+ return paginatedResults;
184
+ }
185
+ catch (error) {
186
+ const duration = Date.now() - startTime;
187
+ ErrorHandler.logError('UnifiedIndexManager.search', error, { query: normalizedSearchOptions, duration });
188
+ throw ErrorHandler.wrapError(error, 'Failed to perform unified portfolio search', ErrorCategory.SYSTEM_ERROR);
189
+ }
190
+ }
191
+ /**
192
+ * Find element by name across all portfolios
193
+ */
194
+ async findByName(name, options = {}) {
195
+ try {
196
+ const searchOptions = {
197
+ query: name,
198
+ includeLocal: options.includeLocal ?? true,
199
+ includeGitHub: options.includeGitHub ?? true,
200
+ includeCollection: options.includeCollection ?? false,
201
+ pageSize: 1,
202
+ ...options
203
+ };
204
+ const results = await this.search(searchOptions);
205
+ // Return exact name match first, then best match
206
+ const exactMatch = results.find(result => result.entry.name.toLowerCase() === name.toLowerCase());
207
+ return exactMatch?.entry || results[0]?.entry || null;
208
+ }
209
+ catch (error) {
210
+ ErrorHandler.logError('UnifiedIndexManager.findByName', error, { name });
211
+ return null;
212
+ }
213
+ }
214
+ /**
215
+ * Get elements by type from all portfolios
216
+ */
217
+ async getElementsByType(elementType, options = {}) {
218
+ try {
219
+ const searchOptions = {
220
+ query: '', // Empty query to get all elements
221
+ elementType,
222
+ includeLocal: options.includeLocal ?? true,
223
+ includeGitHub: options.includeGitHub ?? true,
224
+ includeCollection: options.includeCollection ?? false,
225
+ pageSize: 1000, // Large page size to get all
226
+ ...options
227
+ };
228
+ const results = await this.getAllElementsByType(elementType, searchOptions);
229
+ return this.deduplicateEntries(results.map(r => r.entry));
230
+ }
231
+ catch (error) {
232
+ ErrorHandler.logError('UnifiedIndexManager.getElementsByType', error, { elementType });
233
+ return [];
234
+ }
235
+ }
236
+ /**
237
+ * Check for duplicates across all sources
238
+ */
239
+ async checkDuplicates(name) {
240
+ try {
241
+ const searchOptions = {
242
+ query: name,
243
+ includeLocal: true,
244
+ includeGitHub: true,
245
+ includeCollection: true,
246
+ pageSize: 100
247
+ };
248
+ const results = await this.search(searchOptions);
249
+ const duplicateMap = new Map();
250
+ for (const result of results) {
251
+ const key = `${result.entry.elementType}:${result.entry.name.toLowerCase()}`;
252
+ if (!duplicateMap.has(key)) {
253
+ duplicateMap.set(key, {
254
+ name: result.entry.name,
255
+ elementType: result.entry.elementType,
256
+ sources: [],
257
+ hasVersionConflict: false
258
+ });
259
+ }
260
+ const duplicate = duplicateMap.get(key);
261
+ duplicate.sources.push({
262
+ source: result.source,
263
+ version: result.entry.version,
264
+ lastModified: result.entry.lastModified,
265
+ path: this.getPathFromEntry(result.entry)
266
+ });
267
+ }
268
+ // Filter to only items with multiple sources and check version conflicts
269
+ const actualDuplicates = Array.from(duplicateMap.values())
270
+ .filter(item => item.sources.length > 1)
271
+ .map(item => {
272
+ const versionConflict = this.detectVersionConflict(item.sources);
273
+ return {
274
+ ...item,
275
+ hasVersionConflict: !!versionConflict,
276
+ versionConflict
277
+ };
278
+ });
279
+ return actualDuplicates;
280
+ }
281
+ catch (error) {
282
+ ErrorHandler.logError('UnifiedIndexManager.checkDuplicates', error, { name });
283
+ return [];
284
+ }
285
+ }
286
+ /**
287
+ * Get version comparison across all sources
288
+ */
289
+ async getVersionComparison(name) {
290
+ try {
291
+ const duplicates = await this.checkDuplicates(name);
292
+ if (duplicates.length === 0) {
293
+ return null;
294
+ }
295
+ const duplicate = duplicates[0];
296
+ const versions = {};
297
+ // Build version info
298
+ for (const source of duplicate.sources) {
299
+ if (source.source === 'local') {
300
+ versions.local = {
301
+ version: source.version || 'unknown',
302
+ lastModified: source.lastModified,
303
+ path: source.path || 'unknown'
304
+ };
305
+ }
306
+ else if (source.source === 'github') {
307
+ versions.github = {
308
+ version: source.version || 'unknown',
309
+ lastModified: source.lastModified,
310
+ path: source.path || 'unknown'
311
+ };
312
+ }
313
+ else if (source.source === 'collection') {
314
+ versions.collection = {
315
+ version: source.version || 'unknown',
316
+ lastModified: source.lastModified,
317
+ path: source.path || 'unknown'
318
+ };
319
+ }
320
+ }
321
+ // Determine recommendation
322
+ const recommendation = this.determineVersionRecommendation(versions);
323
+ return {
324
+ name: duplicate.name,
325
+ elementType: duplicate.elementType,
326
+ versions,
327
+ recommended: recommendation,
328
+ updateAvailable: recommendation.source !== 'local' && !!versions.local,
329
+ updateFrom: recommendation.source !== 'local' ? recommendation.source : undefined
330
+ };
331
+ }
332
+ catch (error) {
333
+ ErrorHandler.logError('UnifiedIndexManager.getVersionComparison', error, { name });
334
+ return null;
335
+ }
336
+ }
337
+ /**
338
+ * Get comprehensive statistics across all sources
339
+ */
340
+ async getStats() {
341
+ try {
342
+ const [localStats, githubStats, collectionStats] = await Promise.allSettled([
343
+ this.getLocalStats(),
344
+ this.getGitHubStats(),
345
+ this.getCollectionStats()
346
+ ]);
347
+ const local = localStats.status === 'fulfilled' ? localStats.value : {
348
+ totalElements: 0,
349
+ elementsByType: {},
350
+ lastBuilt: null,
351
+ isStale: true
352
+ };
353
+ const github = githubStats.status === 'fulfilled' ? githubStats.value : {
354
+ totalElements: 0,
355
+ elementsByType: {},
356
+ lastFetched: null,
357
+ isStale: true
358
+ };
359
+ const collection = collectionStats.status === 'fulfilled' ? collectionStats.value : {
360
+ totalElements: 0,
361
+ elementsByType: {},
362
+ lastFetched: null,
363
+ isStale: true
364
+ };
365
+ // Calculate combined statistics
366
+ const totalElements = local.totalElements + github.totalElements + collection.totalElements;
367
+ const duplicatesCount = await this.calculateDuplicatesCount();
368
+ const uniqueElements = totalElements - duplicatesCount;
369
+ return {
370
+ local,
371
+ github,
372
+ collection,
373
+ combined: {
374
+ totalElements,
375
+ uniqueElements,
376
+ duplicates: duplicatesCount
377
+ },
378
+ performance: {
379
+ averageSearchTime: this.getPerformanceStats().searchStats.averageTime || 0,
380
+ cacheHitRate: this.getPerformanceStats().searchStats.cacheHitRate || 0,
381
+ lastOptimized: null
382
+ }
383
+ };
384
+ }
385
+ catch (error) {
386
+ ErrorHandler.logError('UnifiedIndexManager.getStats', error);
387
+ throw error;
388
+ }
389
+ }
390
+ /**
391
+ * Invalidate caches after user actions with performance monitoring
392
+ */
393
+ invalidateAfterAction(action) {
394
+ logger.info('Invalidating unified portfolio caches after user action', { action });
395
+ // Clear result and index caches
396
+ this.resultCache.clear();
397
+ this.indexCache.clear();
398
+ // Invalidate local cache
399
+ this.localIndexManager.rebuildIndex().catch(error => {
400
+ logger.warn('Failed to rebuild local index after action', {
401
+ action,
402
+ error: error instanceof Error ? error.message : String(error)
403
+ });
404
+ });
405
+ // Invalidate GitHub cache
406
+ this.githubIndexer.invalidateAfterAction(action);
407
+ // Invalidate collection cache
408
+ this.collectionIndexCache.clearCache().catch(error => {
409
+ logger.warn('Failed to clear collection cache after action', {
410
+ action,
411
+ error: error instanceof Error ? error.message : String(error)
412
+ });
413
+ });
414
+ // Trigger garbage collection if memory usage is high
415
+ this.triggerMemoryCleanup();
416
+ }
417
+ /**
418
+ * Force rebuild of all indexes with performance optimization
419
+ */
420
+ async rebuildAll() {
421
+ const startTime = Date.now();
422
+ logger.info('Rebuilding all portfolio indexes with optimization...');
423
+ try {
424
+ // Clear all caches
425
+ this.resultCache.clear();
426
+ this.indexCache.clear();
427
+ // Reset performance counters
428
+ this.performanceMonitor.reset();
429
+ // Rebuild in parallel with memory monitoring
430
+ const rebuildPromises = [
431
+ this.localIndexManager.rebuildIndex(),
432
+ this.githubIndexer.clearCache(),
433
+ this.collectionIndexCache.clearCache()
434
+ ];
435
+ await Promise.all(rebuildPromises);
436
+ // Trigger cleanup
437
+ this.triggerMemoryCleanup();
438
+ const duration = Date.now() - startTime;
439
+ logger.info('All portfolio indexes rebuilt successfully', {
440
+ duration: `${duration}ms`,
441
+ memoryUsageMB: process.memoryUsage().heapUsed / (1024 * 1024)
442
+ });
443
+ }
444
+ catch (error) {
445
+ ErrorHandler.logError('UnifiedIndexManager.rebuildAll', error);
446
+ throw error;
447
+ }
448
+ }
449
+ // =====================================================
450
+ // PRIVATE HELPER METHODS
451
+ // =====================================================
452
+ /**
453
+ * Search with fallback strategies for resilient operation
454
+ */
455
+ async searchWithFallback(source, query, options) {
456
+ const startTime = Date.now();
457
+ try {
458
+ let results = [];
459
+ switch (source) {
460
+ case 'local':
461
+ results = await this.searchLocal(query, options);
462
+ break;
463
+ case 'github':
464
+ results = await this.searchGitHub(query, options);
465
+ break;
466
+ case 'collection':
467
+ results = await this.searchCollection(query, options);
468
+ break;
469
+ }
470
+ logger.debug(`${source} search completed in ${Date.now() - startTime}ms with ${results.length} results`);
471
+ return results;
472
+ }
473
+ catch (error) {
474
+ logger.debug(`${source} search failed, attempting fallback`, {
475
+ error: error instanceof Error ? error.message : String(error)
476
+ });
477
+ // Fallback strategies
478
+ return await this.handleSearchFallback(source, query, options, error);
479
+ }
480
+ }
481
+ /**
482
+ * Handle search fallback strategies
483
+ */
484
+ async handleSearchFallback(source, query, options, originalError) {
485
+ try {
486
+ switch (source) {
487
+ case 'local':
488
+ // Try to use stale local index
489
+ logger.debug('Attempting to use stale local index');
490
+ return await this.searchLocalStale(query, options);
491
+ case 'github':
492
+ // Try cached GitHub data
493
+ logger.debug('Attempting to use cached GitHub data');
494
+ return await this.searchGitHubCached(query, options);
495
+ case 'collection':
496
+ // Try cached collection data
497
+ logger.debug('Attempting to use cached collection data');
498
+ return await this.searchCollectionCached(query, options);
499
+ default:
500
+ return [];
501
+ }
502
+ }
503
+ catch (fallbackError) {
504
+ logger.warn(`All fallback strategies failed for ${source}`, {
505
+ originalError: originalError instanceof Error ? originalError.message : String(originalError),
506
+ fallbackError: fallbackError instanceof Error ? fallbackError.message : String(fallbackError)
507
+ });
508
+ return [];
509
+ }
510
+ }
511
+ /**
512
+ * Search local portfolio
513
+ */
514
+ async searchLocal(query, options) {
515
+ const localOptions = this.convertToLocalOptions(options);
516
+ const results = await this.localIndexManager.search(query, localOptions);
517
+ return results.map(result => ({
518
+ source: 'local',
519
+ entry: this.convertLocalEntry(result.entry),
520
+ matchType: result.matchType,
521
+ score: result.score,
522
+ version: result.entry.metadata.version
523
+ }));
524
+ }
525
+ /**
526
+ * Search local with stale data fallback
527
+ */
528
+ async searchLocalStale(query, options) {
529
+ try {
530
+ // Try to get any local data, even stale
531
+ const localOptions = this.convertToLocalOptions(options);
532
+ const results = await this.localIndexManager.search(query, localOptions);
533
+ return results.map(result => ({
534
+ source: 'local',
535
+ entry: this.convertLocalEntry(result.entry),
536
+ matchType: result.matchType,
537
+ score: result.score * 0.8, // Reduce score for stale data
538
+ version: result.entry.metadata.version
539
+ }));
540
+ }
541
+ catch {
542
+ return [];
543
+ }
544
+ }
545
+ /**
546
+ * Search GitHub portfolio
547
+ */
548
+ async searchGitHub(query, options) {
549
+ try {
550
+ const githubIndex = await this.githubIndexer.getIndex();
551
+ const results = [];
552
+ const queryLower = query.toLowerCase();
553
+ const queryTokens = queryLower.split(/\s+/).filter(token => token.length > 0);
554
+ if (queryTokens.length === 0 && query.trim() !== '') {
555
+ return results;
556
+ }
557
+ // Search across all GitHub elements
558
+ for (const [elementType, entries] of githubIndex.elements) {
559
+ // Filter by element type if specified
560
+ if (options.elementType && elementType !== options.elementType) {
561
+ continue;
562
+ }
563
+ for (const entry of entries) {
564
+ const score = this.calculateGitHubMatchScore(entry, queryTokens, query);
565
+ if (score > 0 || query.trim() === '') {
566
+ results.push({
567
+ source: 'github',
568
+ entry: this.convertGitHubEntry(entry),
569
+ matchType: this.determineMatchType(entry, queryTokens),
570
+ score: query.trim() === '' ? 1 : score, // Default score for empty query
571
+ version: entry.version
572
+ });
573
+ }
574
+ }
575
+ }
576
+ return results.sort((a, b) => b.score - a.score);
577
+ }
578
+ catch (error) {
579
+ logger.debug('GitHub search failed', {
580
+ error: error instanceof Error ? error.message : String(error)
581
+ });
582
+ throw error; // Re-throw to trigger fallback
583
+ }
584
+ }
585
+ /**
586
+ * Search GitHub with cached data fallback
587
+ */
588
+ async searchGitHubCached(query, options) {
589
+ try {
590
+ // Try to use stale GitHub data
591
+ const cacheStats = this.githubIndexer.getCacheStats();
592
+ if (!cacheStats.isStale) {
593
+ return await this.searchGitHub(query, options);
594
+ }
595
+ // Use stale data with reduced scores
596
+ const results = await this.searchGitHub(query, options);
597
+ return results.map(result => ({
598
+ ...result,
599
+ score: result.score * 0.7 // Reduce score for stale data
600
+ }));
601
+ }
602
+ catch {
603
+ return [];
604
+ }
605
+ }
606
+ /**
607
+ * Search collection portfolio
608
+ */
609
+ async searchCollection(query, options) {
610
+ try {
611
+ const collectionIndex = await this.collectionIndexCache.getIndex();
612
+ const results = [];
613
+ const queryLower = query.toLowerCase();
614
+ const queryTokens = queryLower.split(/\s+/).filter(token => token.length > 0);
615
+ if (queryTokens.length === 0 && query.trim() !== '') {
616
+ return results;
617
+ }
618
+ // Search across all collection elements
619
+ for (const [elementType, entries] of Object.entries(collectionIndex.index)) {
620
+ // Filter by element type if specified
621
+ if (options.elementType && elementType !== options.elementType.toString()) {
622
+ continue;
623
+ }
624
+ for (const entry of entries) {
625
+ const score = this.calculateCollectionMatchScore(entry, queryTokens, query);
626
+ if (score > 0 || query.trim() === '') {
627
+ results.push({
628
+ source: 'collection',
629
+ entry: this.convertCollectionEntry(entry, elementType),
630
+ matchType: this.determineCollectionMatchType(entry, queryTokens),
631
+ score: query.trim() === '' ? 1 : score, // Default score for empty query
632
+ version: entry.version
633
+ });
634
+ }
635
+ }
636
+ }
637
+ return results.sort((a, b) => b.score - a.score);
638
+ }
639
+ catch (error) {
640
+ logger.debug('Collection search failed', {
641
+ error: error instanceof Error ? error.message : String(error)
642
+ });
643
+ throw error; // Re-throw to trigger fallback
644
+ }
645
+ }
646
+ /**
647
+ * Search collection with cached data fallback
648
+ */
649
+ async searchCollectionCached(query, options) {
650
+ try {
651
+ // Try to use stale collection data
652
+ const cacheStats = this.collectionIndexCache.getCacheStats();
653
+ if (cacheStats.isValid) {
654
+ return await this.searchCollection(query, options);
655
+ }
656
+ // Use stale data with reduced scores
657
+ const results = await this.searchCollection(query, options);
658
+ return results.map(result => ({
659
+ ...result,
660
+ score: result.score * 0.6 // Reduce score for stale collection data
661
+ }));
662
+ }
663
+ catch {
664
+ return [];
665
+ }
666
+ }
667
+ /**
668
+ * Process search results with advanced features
669
+ */
670
+ async processSearchResults(results, options) {
671
+ // Apply smart ranking
672
+ const rankedResults = this.applySmartRanking(results, options);
673
+ // Detect duplicates and version conflicts
674
+ const processedResults = await this.detectDuplicatesAndConflicts(rankedResults);
675
+ // Apply sorting
676
+ const sortedResults = this.applySorting(processedResults, options.sortBy || 'relevance', options.query);
677
+ return sortedResults;
678
+ }
679
+ /**
680
+ * Apply smart result ranking
681
+ */
682
+ applySmartRanking(results, options) {
683
+ return results.map(result => {
684
+ let adjustedScore = result.score;
685
+ // No location-based scoring - score should be based on relevance only
686
+ // Source location doesn't affect the intrinsic value of an element
687
+ // Consider version freshness (newer versions get small bonus)
688
+ if (result.version && result.version !== 'unknown') {
689
+ const versionParts = result.version.split('.');
690
+ if (versionParts.length >= 2) {
691
+ const major = parseInt(versionParts[0]) || 0;
692
+ const minor = parseInt(versionParts[1]) || 0;
693
+ adjustedScore += (major * 0.1) + (minor * 0.01);
694
+ }
695
+ }
696
+ // Boost exact matches
697
+ if (result.entry.name.toLowerCase() === options.query.toLowerCase()) {
698
+ adjustedScore *= 2.0;
699
+ }
700
+ return {
701
+ ...result,
702
+ score: adjustedScore
703
+ };
704
+ });
705
+ }
706
+ /**
707
+ * Detect duplicates and version conflicts
708
+ */
709
+ async detectDuplicatesAndConflicts(results) {
710
+ const nameMap = new Map();
711
+ // Group by name and element type
712
+ for (const result of results) {
713
+ const key = `${result.entry.elementType}:${result.entry.name.toLowerCase()}`;
714
+ if (!nameMap.has(key)) {
715
+ nameMap.set(key, []);
716
+ }
717
+ nameMap.get(key).push(result);
718
+ }
719
+ const processedResults = [];
720
+ // Process each group
721
+ for (const [key, groupResults] of nameMap) {
722
+ if (groupResults.length === 1) {
723
+ // No duplicates
724
+ processedResults.push(groupResults[0]);
725
+ }
726
+ else {
727
+ // Has duplicates - detect version conflicts
728
+ const versionConflict = this.detectVersionConflictFromResults(groupResults);
729
+ // Mark all results as duplicates and add conflict info
730
+ for (const result of groupResults) {
731
+ processedResults.push({
732
+ ...result,
733
+ isDuplicate: true,
734
+ versionConflict
735
+ });
736
+ }
737
+ }
738
+ }
739
+ return processedResults;
740
+ }
741
+ /**
742
+ * Apply pagination to results
743
+ */
744
+ applyPagination(results, options) {
745
+ const page = options.page || 1;
746
+ const pageSize = options.pageSize || 20;
747
+ const startIndex = (page - 1) * pageSize;
748
+ const endIndex = startIndex + pageSize;
749
+ return results.slice(startIndex, endIndex);
750
+ }
751
+ /**
752
+ * Apply sorting to results
753
+ */
754
+ applySorting(results, sortBy, query) {
755
+ const sorted = [...results];
756
+ switch (sortBy) {
757
+ case 'name':
758
+ sorted.sort((a, b) => a.entry.name.localeCompare(b.entry.name));
759
+ break;
760
+ case 'source':
761
+ sorted.sort((a, b) => {
762
+ const sourceOrder = { 'local': 0, 'github': 1, 'collection': 2 };
763
+ return sourceOrder[a.source] - sourceOrder[b.source];
764
+ });
765
+ break;
766
+ case 'version':
767
+ sorted.sort((a, b) => this.compareVersions(b.version || '0', a.version || '0'));
768
+ break;
769
+ case 'relevance':
770
+ default:
771
+ sorted.sort((a, b) => b.score - a.score);
772
+ break;
773
+ }
774
+ return sorted;
775
+ }
776
+ /**
777
+ * Calculate match score for GitHub entries
778
+ */
779
+ calculateGitHubMatchScore(entry, queryTokens, query) {
780
+ if (queryTokens.length === 0)
781
+ return 1; // Default score for empty query
782
+ let score = 0;
783
+ const name = entry.name.toLowerCase();
784
+ const description = (entry.description || '').toLowerCase();
785
+ const path = (entry.path || '').toLowerCase();
786
+ // Check name matches
787
+ for (const token of queryTokens) {
788
+ if (name.includes(token)) {
789
+ score += name === token ? 10 : (name.startsWith(token) ? 5 : 2);
790
+ }
791
+ if (description.includes(token)) {
792
+ score += 3;
793
+ }
794
+ if (path.includes(token)) {
795
+ score += 1;
796
+ }
797
+ }
798
+ // Exact query match bonus
799
+ if (name.includes(query.toLowerCase())) {
800
+ score += query.length > 3 ? 15 : 10;
801
+ }
802
+ return score;
803
+ }
804
+ /**
805
+ * Calculate match score for collection entries
806
+ */
807
+ calculateCollectionMatchScore(entry, queryTokens, query) {
808
+ if (queryTokens.length === 0)
809
+ return 1; // Default score for empty query
810
+ let score = 0;
811
+ const name = entry.name.toLowerCase();
812
+ const description = (entry.description || '').toLowerCase();
813
+ const path = (entry.path || '').toLowerCase();
814
+ const tags = entry.tags.map(tag => tag.toLowerCase()).join(' ');
815
+ // Check matches across all fields
816
+ for (const token of queryTokens) {
817
+ if (name.includes(token)) {
818
+ score += name === token ? 10 : (name.startsWith(token) ? 5 : 2);
819
+ }
820
+ if (description.includes(token)) {
821
+ score += 3;
822
+ }
823
+ if (path.includes(token)) {
824
+ score += 1;
825
+ }
826
+ if (tags.includes(token)) {
827
+ score += 4;
828
+ }
829
+ }
830
+ // Exact query match bonus
831
+ if (name.includes(query.toLowerCase())) {
832
+ score += query.length > 3 ? 15 : 10;
833
+ }
834
+ return score;
835
+ }
836
+ /**
837
+ * Get all elements by type across sources
838
+ */
839
+ async getAllElementsByType(elementType, options) {
840
+ const promises = [];
841
+ if (options.includeLocal) {
842
+ promises.push(this.getLocalElementsByType(elementType));
843
+ }
844
+ if (options.includeGitHub) {
845
+ promises.push(this.getGitHubElementsByType(elementType));
846
+ }
847
+ if (options.includeCollection) {
848
+ promises.push(this.getCollectionElementsByType(elementType));
849
+ }
850
+ const results = await Promise.allSettled(promises);
851
+ const allResults = [];
852
+ results.forEach(result => {
853
+ if (result.status === 'fulfilled') {
854
+ allResults.push(...result.value);
855
+ }
856
+ });
857
+ return allResults;
858
+ }
859
+ /**
860
+ * Get local elements by type
861
+ */
862
+ async getLocalElementsByType(elementType) {
863
+ try {
864
+ const elements = await this.localIndexManager.getElementsByType(elementType);
865
+ return elements.map(entry => ({
866
+ source: 'local',
867
+ entry: this.convertLocalEntry(entry),
868
+ matchType: 'type',
869
+ score: 1,
870
+ version: entry.metadata.version
871
+ }));
872
+ }
873
+ catch {
874
+ return [];
875
+ }
876
+ }
877
+ /**
878
+ * Get GitHub elements by type
879
+ */
880
+ async getGitHubElementsByType(elementType) {
881
+ try {
882
+ const githubIndex = await this.githubIndexer.getIndex();
883
+ const entries = githubIndex.elements.get(elementType) || [];
884
+ return entries.map(entry => ({
885
+ source: 'github',
886
+ entry: this.convertGitHubEntry(entry),
887
+ matchType: 'type',
888
+ score: 1,
889
+ version: entry.version
890
+ }));
891
+ }
892
+ catch {
893
+ return [];
894
+ }
895
+ }
896
+ /**
897
+ * Get collection elements by type
898
+ */
899
+ async getCollectionElementsByType(elementType) {
900
+ try {
901
+ const collectionIndex = await this.collectionIndexCache.getIndex();
902
+ const entries = collectionIndex.index[elementType.toString()] || [];
903
+ return entries.map(entry => ({
904
+ source: 'collection',
905
+ entry: this.convertCollectionEntry(entry, elementType.toString()),
906
+ matchType: 'type',
907
+ score: 1,
908
+ version: entry.version
909
+ }));
910
+ }
911
+ catch {
912
+ return [];
913
+ }
914
+ }
915
+ /**
916
+ * Get local portfolio statistics
917
+ */
918
+ async getLocalStats() {
919
+ return await this.localIndexManager.getStats();
920
+ }
921
+ /**
922
+ * Get GitHub portfolio statistics
923
+ */
924
+ async getGitHubStats() {
925
+ const cacheStats = this.githubIndexer.getCacheStats();
926
+ const githubIndex = await this.githubIndexer.getIndex();
927
+ const elementsByType = {};
928
+ for (const elementType of Object.values(ElementType)) {
929
+ elementsByType[elementType] = (githubIndex.elements.get(elementType) || []).length;
930
+ }
931
+ return {
932
+ totalElements: githubIndex.totalElements,
933
+ elementsByType,
934
+ lastFetched: cacheStats.lastFetch,
935
+ isStale: cacheStats.isStale,
936
+ username: githubIndex.username,
937
+ repository: githubIndex.repository
938
+ };
939
+ }
940
+ /**
941
+ * Get collection portfolio statistics
942
+ */
943
+ async getCollectionStats() {
944
+ const cacheStats = this.collectionIndexCache.getCacheStats();
945
+ const collectionIndex = await this.collectionIndexCache.getIndex();
946
+ const elementsByType = {};
947
+ for (const [elementType, entries] of Object.entries(collectionIndex.index)) {
948
+ elementsByType[elementType] = entries.length;
949
+ }
950
+ return {
951
+ totalElements: collectionIndex.total_elements,
952
+ elementsByType,
953
+ lastFetched: cacheStats.hasCache ? new Date(Date.now() - cacheStats.age) : null,
954
+ isStale: !cacheStats.isValid,
955
+ version: collectionIndex.version
956
+ };
957
+ }
958
+ /**
959
+ * Calculate duplicates count across all sources
960
+ */
961
+ async calculateDuplicatesCount() {
962
+ try {
963
+ // This is a placeholder - actual implementation would need optimization
964
+ // For now, return 0 to avoid the expensive operation during stats calculation
965
+ return 0;
966
+ }
967
+ catch {
968
+ return 0;
969
+ }
970
+ }
971
+ /**
972
+ * Convert local index entry to unified format
973
+ */
974
+ convertLocalEntry(entry) {
975
+ return {
976
+ name: entry.metadata.name,
977
+ description: entry.metadata.description,
978
+ version: entry.metadata.version,
979
+ author: entry.metadata.author,
980
+ elementType: entry.elementType,
981
+ lastModified: entry.lastModified,
982
+ source: 'local',
983
+ localFilePath: entry.filePath,
984
+ filename: entry.filename,
985
+ tags: entry.metadata.tags,
986
+ keywords: entry.metadata.keywords,
987
+ triggers: entry.metadata.triggers,
988
+ category: entry.metadata.category
989
+ };
990
+ }
991
+ /**
992
+ * Convert GitHub index entry to unified format
993
+ */
994
+ convertGitHubEntry(entry) {
995
+ return {
996
+ name: entry.name,
997
+ description: entry.description,
998
+ version: entry.version,
999
+ author: entry.author,
1000
+ elementType: entry.elementType,
1001
+ lastModified: entry.lastModified,
1002
+ source: 'github',
1003
+ githubPath: entry.path,
1004
+ githubSha: entry.sha,
1005
+ githubHtmlUrl: entry.htmlUrl,
1006
+ githubDownloadUrl: entry.downloadUrl,
1007
+ githubSize: entry.size
1008
+ };
1009
+ }
1010
+ /**
1011
+ * Convert collection index entry to unified format
1012
+ */
1013
+ convertCollectionEntry(entry, elementType) {
1014
+ return {
1015
+ name: entry.name,
1016
+ description: entry.description,
1017
+ version: entry.version,
1018
+ author: entry.author,
1019
+ elementType: this.mapStringToElementType(elementType),
1020
+ lastModified: new Date(entry.created),
1021
+ source: 'collection',
1022
+ collectionPath: entry.path,
1023
+ collectionSha: entry.sha,
1024
+ collectionTags: entry.tags,
1025
+ collectionCategory: entry.category,
1026
+ collectionLicense: entry.license
1027
+ };
1028
+ }
1029
+ /**
1030
+ * Map string to ElementType enum
1031
+ */
1032
+ mapStringToElementType(elementType) {
1033
+ // Handle mapping from collection element types to our ElementType enum
1034
+ switch (elementType.toLowerCase()) {
1035
+ case 'personas':
1036
+ return ElementType.PERSONA;
1037
+ case 'skills':
1038
+ return ElementType.SKILL;
1039
+ case 'agents':
1040
+ return ElementType.AGENT;
1041
+ case 'prompts':
1042
+ case 'templates':
1043
+ return ElementType.TEMPLATE; // Map prompts and templates to TEMPLATE
1044
+ case 'tools':
1045
+ return ElementType.SKILL; // Map tools to SKILL as fallback
1046
+ case 'ensembles':
1047
+ return ElementType.ENSEMBLE;
1048
+ case 'memories':
1049
+ return ElementType.MEMORY;
1050
+ default:
1051
+ return ElementType.SKILL; // Default fallback
1052
+ }
1053
+ }
1054
+ /**
1055
+ * Convert unified search options to local search options
1056
+ */
1057
+ convertToLocalOptions(options) {
1058
+ return {
1059
+ elementType: options.elementType,
1060
+ maxResults: options.pageSize || 20
1061
+ };
1062
+ }
1063
+ /**
1064
+ * Determine match type for GitHub entries
1065
+ */
1066
+ determineMatchType(entry, queryTokens) {
1067
+ const name = entry.name.toLowerCase();
1068
+ const description = (entry.description || '').toLowerCase();
1069
+ // Check what matched
1070
+ for (const token of queryTokens) {
1071
+ if (name.includes(token)) {
1072
+ return name === token ? 'exact_name' : 'name';
1073
+ }
1074
+ if (description.includes(token)) {
1075
+ return 'description';
1076
+ }
1077
+ }
1078
+ return 'content';
1079
+ }
1080
+ /**
1081
+ * Determine match type for collection entries
1082
+ */
1083
+ determineCollectionMatchType(entry, queryTokens) {
1084
+ const name = entry.name.toLowerCase();
1085
+ const description = (entry.description || '').toLowerCase();
1086
+ const tags = entry.tags.map(tag => tag.toLowerCase()).join(' ');
1087
+ // Check what matched
1088
+ for (const token of queryTokens) {
1089
+ if (name.includes(token)) {
1090
+ return name === token ? 'exact_name' : 'name';
1091
+ }
1092
+ if (description.includes(token)) {
1093
+ return 'description';
1094
+ }
1095
+ if (tags.includes(token)) {
1096
+ return 'tag';
1097
+ }
1098
+ }
1099
+ return 'content';
1100
+ }
1101
+ /**
1102
+ * Get path from unified entry
1103
+ */
1104
+ getPathFromEntry(entry) {
1105
+ switch (entry.source) {
1106
+ case 'local':
1107
+ return entry.localFilePath || entry.filename || 'unknown';
1108
+ case 'github':
1109
+ return entry.githubPath || 'unknown';
1110
+ case 'collection':
1111
+ return entry.collectionPath || 'unknown';
1112
+ default:
1113
+ return 'unknown';
1114
+ }
1115
+ }
1116
+ /**
1117
+ * Detect version conflict from sources
1118
+ */
1119
+ detectVersionConflict(sources) {
1120
+ const versions = new Map();
1121
+ for (const source of sources) {
1122
+ if (source.version && source.version !== 'unknown') {
1123
+ versions.set(source.version, source.source);
1124
+ }
1125
+ }
1126
+ if (versions.size <= 1) {
1127
+ return undefined; // No conflict if all versions are the same or missing
1128
+ }
1129
+ // Build version conflict info
1130
+ const versionConflict = {
1131
+ recommended: 'local',
1132
+ reason: 'Multiple versions detected'
1133
+ };
1134
+ for (const source of sources) {
1135
+ if (source.version) {
1136
+ versionConflict[source.source] = source.version;
1137
+ }
1138
+ }
1139
+ // Determine recommendation
1140
+ const recommendation = this.determineVersionRecommendationFromSources(sources);
1141
+ versionConflict.recommended = recommendation.source;
1142
+ versionConflict.reason = recommendation.reason;
1143
+ return versionConflict;
1144
+ }
1145
+ /**
1146
+ * Detect version conflict from search results
1147
+ */
1148
+ detectVersionConflictFromResults(results) {
1149
+ const sources = results.map(result => ({
1150
+ source: result.source,
1151
+ version: result.version,
1152
+ lastModified: result.entry.lastModified,
1153
+ path: this.getPathFromEntry(result.entry)
1154
+ }));
1155
+ return this.detectVersionConflict(sources);
1156
+ }
1157
+ /**
1158
+ * Determine version recommendation from version info
1159
+ */
1160
+ determineVersionRecommendation(versions) {
1161
+ // Prefer local if available and not too old
1162
+ if (versions.local) {
1163
+ const localAge = Date.now() - versions.local.lastModified.getTime();
1164
+ const sevenDays = 7 * 24 * 60 * 60 * 1000;
1165
+ if (localAge < sevenDays) {
1166
+ return { source: 'local', reason: 'Local version is recent and authoritative' };
1167
+ }
1168
+ }
1169
+ // Compare versions if available
1170
+ const versionEntries = Object.entries(versions).filter(([_, info]) => info?.version);
1171
+ if (versionEntries.length > 1) {
1172
+ // Find highest version
1173
+ let highest = {
1174
+ source: 'local',
1175
+ version: '0.0.0'
1176
+ };
1177
+ for (const [source, info] of versionEntries) {
1178
+ if (info && this.compareVersions(info.version, highest.version) > 0) {
1179
+ highest = {
1180
+ source: source,
1181
+ version: info.version
1182
+ };
1183
+ }
1184
+ }
1185
+ return { source: highest.source, reason: `Highest version (${highest.version})` };
1186
+ }
1187
+ // Fallback to most recent
1188
+ let mostRecent = {
1189
+ source: 'local',
1190
+ date: new Date(0)
1191
+ };
1192
+ for (const [source, info] of Object.entries(versions)) {
1193
+ if (info && info.lastModified > mostRecent.date) {
1194
+ mostRecent = {
1195
+ source: source,
1196
+ date: info.lastModified
1197
+ };
1198
+ }
1199
+ }
1200
+ return { source: mostRecent.source, reason: 'Most recently modified' };
1201
+ }
1202
+ /**
1203
+ * Determine version recommendation from sources
1204
+ */
1205
+ determineVersionRecommendationFromSources(sources) {
1206
+ // Convert sources to versions format
1207
+ const versions = {};
1208
+ for (const source of sources) {
1209
+ if (source.source === 'local') {
1210
+ versions.local = {
1211
+ version: source.version || 'unknown',
1212
+ lastModified: source.lastModified,
1213
+ path: source.path || 'unknown'
1214
+ };
1215
+ }
1216
+ else if (source.source === 'github') {
1217
+ versions.github = {
1218
+ version: source.version || 'unknown',
1219
+ lastModified: source.lastModified,
1220
+ path: source.path || 'unknown'
1221
+ };
1222
+ }
1223
+ else if (source.source === 'collection') {
1224
+ versions.collection = {
1225
+ version: source.version || 'unknown',
1226
+ lastModified: source.lastModified,
1227
+ path: source.path || 'unknown'
1228
+ };
1229
+ }
1230
+ }
1231
+ return this.determineVersionRecommendation(versions);
1232
+ }
1233
+ /**
1234
+ * Compare semantic versions
1235
+ */
1236
+ compareVersions(a, b) {
1237
+ const parseVersion = (version) => {
1238
+ const parts = version.split('.').map(part => parseInt(part) || 0);
1239
+ return [parts[0] || 0, parts[1] || 0, parts[2] || 0];
1240
+ };
1241
+ const [aMajor, aMinor, aPatch] = parseVersion(a);
1242
+ const [bMajor, bMinor, bPatch] = parseVersion(b);
1243
+ if (aMajor !== bMajor)
1244
+ return aMajor - bMajor;
1245
+ if (aMinor !== bMinor)
1246
+ return aMinor - bMinor;
1247
+ return aPatch - bPatch;
1248
+ }
1249
+ /**
1250
+ * Remove duplicate results based on name and type
1251
+ */
1252
+ deduplicateResults(results) {
1253
+ const seen = new Set();
1254
+ const deduplicated = [];
1255
+ for (const result of results) {
1256
+ const key = `${result.entry.elementType}:${result.entry.name.toLowerCase()}`;
1257
+ if (!seen.has(key)) {
1258
+ seen.add(key);
1259
+ deduplicated.push(result);
1260
+ }
1261
+ }
1262
+ return deduplicated;
1263
+ }
1264
+ /**
1265
+ * Remove duplicate entries based on name and type
1266
+ */
1267
+ deduplicateEntries(entries) {
1268
+ const seen = new Set();
1269
+ const deduplicated = [];
1270
+ for (const entry of entries) {
1271
+ const key = `${entry.elementType}:${entry.name.toLowerCase()}`;
1272
+ if (!seen.has(key)) {
1273
+ seen.add(key);
1274
+ deduplicated.push(entry);
1275
+ }
1276
+ }
1277
+ return deduplicated;
1278
+ }
1279
+ // =====================================================
1280
+ // PERFORMANCE MONITORING AND OPTIMIZATION
1281
+ // =====================================================
1282
+ /**
1283
+ * Stream search results for better performance with large datasets
1284
+ */
1285
+ async streamSearch(options) {
1286
+ const { query, cursor, maxResults = 1000 } = options;
1287
+ const startTime = Date.now();
1288
+ logger.debug('Starting streaming search', { query: query.substring(0, 50), cursor, maxResults });
1289
+ const results = [];
1290
+ const sources = this.getEnabledSources(options);
1291
+ // Process sources in sequence for memory efficiency
1292
+ for (const source of sources) {
1293
+ if (results.length >= maxResults) {
1294
+ break;
1295
+ }
1296
+ try {
1297
+ const sourceResults = await this.searchWithFallback(source, query, {
1298
+ ...options,
1299
+ pageSize: Math.min(this.BATCH_SIZE, maxResults - results.length)
1300
+ });
1301
+ results.push(...sourceResults);
1302
+ // Add cursor information for pagination
1303
+ sourceResults.forEach((result, index) => {
1304
+ result.cursor = this.generateCursor(source, index);
1305
+ });
1306
+ }
1307
+ catch (error) {
1308
+ logger.warn(`Streaming search failed for source ${source}`, {
1309
+ error: error instanceof Error ? error.message : String(error)
1310
+ });
1311
+ }
1312
+ }
1313
+ logger.debug('Streaming search completed', {
1314
+ resultCount: results.length,
1315
+ duration: `${Date.now() - startTime}ms`
1316
+ });
1317
+ return results;
1318
+ }
1319
+ /**
1320
+ * Process search results with memory-efficient batching
1321
+ */
1322
+ async processSearchResultsOptimized(results, options) {
1323
+ if (results.length === 0) {
1324
+ return results;
1325
+ }
1326
+ // Process in batches to avoid memory spikes with large result sets
1327
+ const batchSize = Math.min(this.BATCH_SIZE, results.length);
1328
+ const processedResults = [];
1329
+ for (let i = 0; i < results.length; i += batchSize) {
1330
+ const batch = results.slice(i, i + batchSize);
1331
+ // Apply smart ranking
1332
+ const rankedBatch = this.applySmartRanking(batch, options);
1333
+ // Detect duplicates and conflicts
1334
+ const processedBatch = await this.detectDuplicatesAndConflicts(rankedBatch);
1335
+ processedResults.push(...processedBatch);
1336
+ // Yield control to prevent blocking
1337
+ if (i % (batchSize * 4) === 0) {
1338
+ await new Promise(resolve => setImmediate(resolve));
1339
+ }
1340
+ }
1341
+ // Apply final sorting
1342
+ return this.applySorting(processedResults, options.sortBy || 'relevance', options.query);
1343
+ }
1344
+ /**
1345
+ * Get enabled search sources
1346
+ */
1347
+ getEnabledSources(options) {
1348
+ const sources = [];
1349
+ if (options.includeLocal !== false)
1350
+ sources.push('local');
1351
+ if (options.includeGitHub !== false)
1352
+ sources.push('github');
1353
+ if (options.includeCollection === true)
1354
+ sources.push('collection');
1355
+ return sources;
1356
+ }
1357
+ /**
1358
+ * Batch sources for concurrent processing
1359
+ */
1360
+ batchSources(sources, batchSize) {
1361
+ const batches = [];
1362
+ for (let i = 0; i < sources.length; i += batchSize) {
1363
+ batches.push(sources.slice(i, i + batchSize));
1364
+ }
1365
+ return batches;
1366
+ }
1367
+ /**
1368
+ * Generate cursor for pagination
1369
+ */
1370
+ generateCursor(source, index) {
1371
+ const timestamp = Date.now();
1372
+ return Buffer.from(`${source}:${index}:${timestamp}`).toString('base64');
1373
+ }
1374
+ /**
1375
+ * Trigger memory cleanup when usage is high
1376
+ */
1377
+ triggerMemoryCleanup() {
1378
+ // Force cache cleanup
1379
+ this.resultCache.cleanup();
1380
+ this.indexCache.cleanup();
1381
+ // Suggest garbage collection
1382
+ if (global.gc) {
1383
+ global.gc();
1384
+ logger.debug('Triggered garbage collection');
1385
+ }
1386
+ }
1387
+ /**
1388
+ * Record search performance metrics
1389
+ */
1390
+ recordSearchMetrics(metrics) {
1391
+ this.performanceMonitor.recordSearch(metrics);
1392
+ // Update cache performance metrics
1393
+ const cacheStats = this.resultCache.getStats();
1394
+ this.performanceMonitor.recordCachePerformance('searchResults', {
1395
+ hitRate: cacheStats.hitRate,
1396
+ avgHitTime: 1, // Placeholder
1397
+ avgMissTime: 5, // Placeholder
1398
+ totalHits: cacheStats.hitCount,
1399
+ totalMisses: cacheStats.missCount,
1400
+ evictions: cacheStats.evictionCount
1401
+ });
1402
+ }
1403
+ /**
1404
+ * Create cache key for search options
1405
+ */
1406
+ createCacheKey(options) {
1407
+ return JSON.stringify({
1408
+ query: options.query,
1409
+ includeLocal: options.includeLocal,
1410
+ includeGitHub: options.includeGitHub,
1411
+ includeCollection: options.includeCollection,
1412
+ elementType: options.elementType,
1413
+ page: options.page,
1414
+ pageSize: options.pageSize,
1415
+ sortBy: options.sortBy,
1416
+ lazyLoad: options.lazyLoad
1417
+ });
1418
+ }
1419
+ /**
1420
+ * Get performance statistics
1421
+ */
1422
+ getPerformanceStats() {
1423
+ return {
1424
+ searchStats: this.performanceMonitor.getSearchStats(),
1425
+ memoryStats: this.performanceMonitor.getMemoryStats(),
1426
+ cacheStats: {
1427
+ searchResults: this.resultCache.getStats(),
1428
+ indexCache: this.indexCache.getStats()
1429
+ },
1430
+ trends: this.performanceMonitor.analyzeTrends()
1431
+ };
1432
+ }
1433
+ }
1434
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UnifiedIndexManager.js","sourceRoot":"","sources":["../../src/portfolio/UnifiedIndexManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Unified Index Manager - Combines local, GitHub, and collection portfolio indexing\n * \n * Features:\n * - Unified search across local, GitHub, and collection portfolios\n * - Intelligent result merging and deduplication\n * - Version conflict detection and resolution\n * - Performance optimization with parallel indexing\n * - Advanced fallback strategies for resilient operation\n * - Comprehensive search capabilities with pagination\n * - Smart result ranking and duplicate detection\n */\n\nimport { PortfolioIndexManager, IndexEntry, SearchResult, SearchOptions } from './PortfolioIndexManager.js';\nimport { GitHubPortfolioIndexer, GitHubIndexEntry, GitHubPortfolioIndex } from './GitHubPortfolioIndexer.js';\nimport { CollectionIndexCache } from '../cache/CollectionIndexCache.js';\nimport { GitHubClient } from '../collection/GitHubClient.js';\nimport { APICache } from '../cache/APICache.js';\nimport { ElementType } from './types.js';\nimport { logger } from '../utils/logger.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\nimport { LRUCache, CacheFactory } from '../cache/LRUCache.js';\nimport { PerformanceMonitor, SearchMetrics } from '../utils/PerformanceMonitor.js';\nimport { IndexEntry as CollectionIndexEntry, CollectionIndex } from '../types/collection.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\n\nexport interface UnifiedSearchOptions {\n  query: string;\n  includeLocal?: boolean;     // default true\n  includeGitHub?: boolean;    // default true\n  includeCollection?: boolean; // default false\n  elementType?: ElementType;\n  page?: number;\n  pageSize?: number;\n  sortBy?: 'relevance' | 'source' | 'name' | 'version';\n  streamResults?: boolean; // Enable result streaming\n  cursor?: string; // For pagination cursor\n  maxResults?: number; // Hard limit on results\n  lazyLoad?: boolean; // Enable lazy loading\n}\n\nexport interface VersionConflict {\n  local?: string;\n  github?: string;\n  collection?: string;\n  recommended: 'local' | 'github' | 'collection';\n  reason: string;\n}\n\nexport interface DuplicateInfo {\n  name: string;\n  elementType: ElementType;\n  sources: Array<{\n    source: 'local' | 'github' | 'collection';\n    version?: string;\n    lastModified: Date;\n    path?: string;\n  }>;\n  hasVersionConflict: boolean;\n  versionConflict?: VersionConflict;\n}\n\nexport interface VersionInfo {\n  name: string;\n  elementType: ElementType;\n  versions: {\n    local?: { version: string; lastModified: Date; path: string };\n    github?: { version: string; lastModified: Date; path: string };\n    collection?: { version: string; lastModified: Date; path: string };\n  };\n  recommended: {\n    source: 'local' | 'github' | 'collection';\n    reason: string;\n  };\n  updateAvailable: boolean;\n  updateFrom?: 'local' | 'github' | 'collection';\n}\n\nexport interface UnifiedIndexEntry {\n  // Common properties\n  name: string;\n  description?: string;\n  version?: string;\n  author?: string;\n  elementType: ElementType;\n  lastModified: Date;\n  \n  // Source information\n  source: 'local' | 'github' | 'collection';\n  \n  // Local properties (when source === 'local')\n  localFilePath?: string;\n  filename?: string;\n  tags?: string[];\n  keywords?: string[];\n  triggers?: string[];\n  category?: string;\n  \n  // GitHub properties (when source === 'github')\n  githubPath?: string;\n  githubSha?: string;\n  githubHtmlUrl?: string;\n  githubDownloadUrl?: string;\n  githubSize?: number;\n  \n  // Collection properties (when source === 'collection')\n  collectionPath?: string;\n  collectionSha?: string;\n  collectionTags?: string[];\n  collectionCategory?: string;\n  collectionLicense?: string;\n}\n\nexport interface UnifiedSearchResult {\n  source: 'local' | 'github' | 'collection';\n  entry: UnifiedIndexEntry;\n  matchType: string;\n  score: number;\n  version?: string;\n  isDuplicate?: boolean;\n  versionConflict?: VersionConflict;\n  cursor?: string; // For streaming pagination\n}\n\nexport interface StreamedSearchResult {\n  results: UnifiedSearchResult[];\n  hasMore: boolean;\n  nextCursor?: string;\n  totalEstimate?: number;\n  processingTimeMs: number;\n}\n\nexport interface UnifiedIndexStats {\n  local: {\n    totalElements: number;\n    elementsByType: Record<ElementType, number>;\n    lastBuilt: Date | null;\n    isStale: boolean;\n  };\n  github: {\n    totalElements: number;\n    elementsByType: Record<ElementType, number>;\n    lastFetched: Date | null;\n    isStale: boolean;\n    username?: string;\n    repository?: string;\n  };\n  collection: {\n    totalElements: number;\n    elementsByType: Record<string, number>;\n    lastFetched: Date | null;\n    isStale: boolean;\n    version?: string;\n  };\n  combined: {\n    totalElements: number;\n    uniqueElements: number;\n    duplicates: number;\n  };\n  performance: {\n    averageSearchTime: number;\n    cacheHitRate: number;\n    lastOptimized: Date | null;\n  };\n}\n\nexport class UnifiedIndexManager {\n  private static instance: UnifiedIndexManager | null = null;\n  \n  private localIndexManager: PortfolioIndexManager;\n  private githubIndexer: GitHubPortfolioIndexer;\n  private collectionIndexCache: CollectionIndexCache;\n  private githubClient: GitHubClient;\n  \n  // Performance monitoring and caching\n  private performanceMonitor: PerformanceMonitor;\n  private resultCache: LRUCache<UnifiedSearchResult[]>;\n  private indexCache: LRUCache<any>;\n  private readonly BATCH_SIZE = 50; // For streaming results\n  private readonly MAX_CONCURRENT_SOURCES = 3;\n  \n  private constructor() {\n    this.localIndexManager = PortfolioIndexManager.getInstance();\n    this.githubIndexer = GitHubPortfolioIndexer.getInstance();\n    \n    // Initialize GitHubClient with required dependencies\n    const apiCache = new APICache();\n    const rateLimitTracker = new Map<string, number[]>();\n    this.githubClient = new GitHubClient(apiCache, rateLimitTracker);\n    this.collectionIndexCache = new CollectionIndexCache(this.githubClient);\n    \n    // Initialize performance monitoring and caching\n    this.performanceMonitor = PerformanceMonitor.getInstance();\n    this.performanceMonitor.startMonitoring();\n    \n    this.resultCache = CacheFactory.createSearchResultCache({\n      maxSize: 200,\n      maxMemoryMB: 15,\n      ttlMs: 5 * 60 * 1000, // 5 minutes\n      onEviction: (key, value) => {\n        logger.debug('Search result cache eviction', { key, resultCount: value.length });\n      }\n    });\n    \n    this.indexCache = CacheFactory.createIndexCache({\n      maxSize: 100,\n      maxMemoryMB: 20,\n      ttlMs: 15 * 60 * 1000, // 15 minutes\n      onEviction: (key, value) => {\n        logger.debug('Index cache eviction', { key });\n      }\n    });\n    \n    logger.debug('UnifiedIndexManager created with performance optimization');\n  }\n\n  public static getInstance(): UnifiedIndexManager {\n    if (!this.instance) {\n      this.instance = new UnifiedIndexManager();\n    }\n    return this.instance;\n  }\n\n  /**\n   * Enhanced search across local, GitHub, and collection portfolios with performance optimization\n   */\n  public async search(searchOptions: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    const startTime = Date.now();\n    const memoryBefore = process.memoryUsage().heapUsed;\n    const { query, includeLocal = true, includeGitHub = true, includeCollection = false } = searchOptions;\n    \n    // Normalize query to prevent Unicode-based attacks\n    const validationResult = UnicodeValidator.normalize(query);\n    const normalizedQuery = validationResult.normalizedContent;\n    \n    // Use normalized query in all subsequent operations\n    const normalizedSearchOptions = {\n      ...searchOptions,\n      query: normalizedQuery\n    };\n    \n    // SECURITY FIX (DMCP-SEC-006): Add audit logging for security monitoring\n    // Log unified search operations for security audit trail\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_FETCH_SUCCESS',\n      severity: 'LOW',\n      source: 'UnifiedIndexManager.search',\n      details: `Unified search performed with query length: ${normalizedQuery.length}, sources: ${JSON.stringify({\n        local: includeLocal,\n        github: includeGitHub,\n        collection: includeCollection\n      })}`\n    });\n    \n    logger.debug('Starting optimized unified portfolio search', normalizedSearchOptions);\n    \n    // Check cache first (use normalized search options)\n    const cacheKey = this.createCacheKey(normalizedSearchOptions);\n    const cached = this.resultCache.get(cacheKey);\n    if (cached) {\n      const duration = Date.now() - startTime;\n      this.recordSearchMetrics({\n        query: normalizedQuery,\n        duration,\n        resultCount: cached.length,\n        sources: this.getEnabledSources(normalizedSearchOptions),\n        cacheHit: true,\n        memoryBefore,\n        memoryAfter: process.memoryUsage().heapUsed,\n        timestamp: new Date()\n      });\n      logger.debug('Using cached search results', { resultCount: cached.length });\n      return cached;\n    }\n    \n    try {\n      // Use streaming search for better performance with large result sets\n      if (normalizedSearchOptions.streamResults) {\n        return await this.streamSearch(normalizedSearchOptions);\n      }\n      \n      // Lazy loading: Only load indices when needed\n      const searchPromises: Promise<UnifiedSearchResult[]>[] = [];\n      const enabledSources = this.getEnabledSources(normalizedSearchOptions);\n      \n      // Limit concurrent source searches for memory efficiency\n      const concurrentLimit = Math.min(this.MAX_CONCURRENT_SOURCES, enabledSources.length);\n      const sourceBatches = this.batchSources(enabledSources, concurrentLimit);\n      \n      const allResults: UnifiedSearchResult[] = [];\n      const sourceCount = { local: 0, github: 0, collection: 0 };\n      \n      // Process sources in batches to control memory usage\n      for (const batch of sourceBatches) {\n        const batchPromises = batch.map(source => \n          this.searchWithFallback(source as 'local' | 'github' | 'collection', normalizedQuery, normalizedSearchOptions)\n        );\n        \n        const batchResults = await Promise.allSettled(batchPromises);\n        \n        batchResults.forEach((result, index) => {\n          const sourceName = batch[index] as 'local' | 'github' | 'collection';\n          if (result.status === 'fulfilled') {\n            sourceCount[sourceName] += result.value.length;\n            allResults.push(...result.value);\n          } else {\n            logger.warn(`Search failed for source ${sourceName}`, {\n              error: result.reason instanceof Error ? result.reason.message : String(result.reason)\n            });\n          }\n        });\n        \n        // Memory check between batches\n        const currentMemory = process.memoryUsage().heapUsed / (1024 * 1024);\n        if (currentMemory > 200) { // 200MB threshold\n          logger.warn('High memory usage during search, triggering cleanup', {\n            memoryMB: currentMemory\n          });\n          this.triggerMemoryCleanup();\n        }\n      }\n      \n      // Apply advanced processing with memory-efficient batching\n      const processedResults = await this.processSearchResultsOptimized(allResults, normalizedSearchOptions);\n      \n      // Apply pagination\n      const paginatedResults = this.applyPagination(processedResults, normalizedSearchOptions);\n      \n      // Cache results with memory limit check\n      if (paginatedResults.length < 1000) { // Don't cache very large result sets\n        this.resultCache.set(cacheKey, paginatedResults);\n      }\n      \n      const duration = Date.now() - startTime;\n      const memoryAfter = process.memoryUsage().heapUsed;\n      \n      this.recordSearchMetrics({\n        query: normalizedQuery,\n        duration,\n        resultCount: paginatedResults.length,\n        sources: enabledSources,\n        cacheHit: false,\n        memoryBefore,\n        memoryAfter,\n        timestamp: new Date()\n      });\n      \n      logger.info('Optimized unified portfolio search completed', {\n        query: normalizedQuery.substring(0, 50),\n        sources: { ...sourceCount, total: allResults.length },\n        finalResults: paginatedResults.length,\n        duration: `${duration}ms`,\n        memoryUsageMB: (memoryAfter - memoryBefore) / (1024 * 1024)\n      });\n      \n      return paginatedResults;\n      \n    } catch (error) {\n      const duration = Date.now() - startTime;\n      ErrorHandler.logError('UnifiedIndexManager.search', error, { query: normalizedSearchOptions, duration });\n      throw ErrorHandler.wrapError(error, 'Failed to perform unified portfolio search', ErrorCategory.SYSTEM_ERROR);\n    }\n  }\n\n  /**\n   * Find element by name across all portfolios\n   */\n  public async findByName(name: string, options: Partial<UnifiedSearchOptions> = {}): Promise<UnifiedIndexEntry | null> {\n    try {\n      const searchOptions: UnifiedSearchOptions = {\n        query: name,\n        includeLocal: options.includeLocal ?? true,\n        includeGitHub: options.includeGitHub ?? true,\n        includeCollection: options.includeCollection ?? false,\n        pageSize: 1,\n        ...options\n      };\n      \n      const results = await this.search(searchOptions);\n      \n      // Return exact name match first, then best match\n      const exactMatch = results.find(result => \n        result.entry.name.toLowerCase() === name.toLowerCase()\n      );\n      \n      return exactMatch?.entry || results[0]?.entry || null;\n      \n    } catch (error) {\n      ErrorHandler.logError('UnifiedIndexManager.findByName', error, { name });\n      return null;\n    }\n  }\n\n  /**\n   * Get elements by type from all portfolios\n   */\n  public async getElementsByType(elementType: ElementType, options: Partial<UnifiedSearchOptions> = {}): Promise<UnifiedIndexEntry[]> {\n    try {\n      const searchOptions: UnifiedSearchOptions = {\n        query: '', // Empty query to get all elements\n        elementType,\n        includeLocal: options.includeLocal ?? true,\n        includeGitHub: options.includeGitHub ?? true,\n        includeCollection: options.includeCollection ?? false,\n        pageSize: 1000, // Large page size to get all\n        ...options\n      };\n      \n      const results = await this.getAllElementsByType(elementType, searchOptions);\n      \n      return this.deduplicateEntries(results.map(r => r.entry));\n      \n    } catch (error) {\n      ErrorHandler.logError('UnifiedIndexManager.getElementsByType', error, { elementType });\n      return [];\n    }\n  }\n  \n  /**\n   * Check for duplicates across all sources\n   */\n  public async checkDuplicates(name: string): Promise<DuplicateInfo[]> {\n    try {\n      const searchOptions: UnifiedSearchOptions = {\n        query: name,\n        includeLocal: true,\n        includeGitHub: true,\n        includeCollection: true,\n        pageSize: 100\n      };\n      \n      const results = await this.search(searchOptions);\n      const duplicateMap = new Map<string, DuplicateInfo>();\n      \n      for (const result of results) {\n        const key = `${result.entry.elementType}:${result.entry.name.toLowerCase()}`;\n        \n        if (!duplicateMap.has(key)) {\n          duplicateMap.set(key, {\n            name: result.entry.name,\n            elementType: result.entry.elementType,\n            sources: [],\n            hasVersionConflict: false\n          });\n        }\n        \n        const duplicate = duplicateMap.get(key)!;\n        duplicate.sources.push({\n          source: result.source,\n          version: result.entry.version,\n          lastModified: result.entry.lastModified,\n          path: this.getPathFromEntry(result.entry)\n        });\n      }\n      \n      // Filter to only items with multiple sources and check version conflicts\n      const actualDuplicates = Array.from(duplicateMap.values())\n        .filter(item => item.sources.length > 1)\n        .map(item => {\n          const versionConflict = this.detectVersionConflict(item.sources);\n          return {\n            ...item,\n            hasVersionConflict: !!versionConflict,\n            versionConflict\n          };\n        });\n      \n      return actualDuplicates;\n      \n    } catch (error) {\n      ErrorHandler.logError('UnifiedIndexManager.checkDuplicates', error, { name });\n      return [];\n    }\n  }\n  \n  /**\n   * Get version comparison across all sources\n   */\n  public async getVersionComparison(name: string): Promise<VersionInfo | null> {\n    try {\n      const duplicates = await this.checkDuplicates(name);\n      \n      if (duplicates.length === 0) {\n        return null;\n      }\n      \n      const duplicate = duplicates[0];\n      const versions: VersionInfo['versions'] = {};\n      \n      // Build version info\n      for (const source of duplicate.sources) {\n        if (source.source === 'local') {\n          versions.local = {\n            version: source.version || 'unknown',\n            lastModified: source.lastModified,\n            path: source.path || 'unknown'\n          };\n        } else if (source.source === 'github') {\n          versions.github = {\n            version: source.version || 'unknown',\n            lastModified: source.lastModified,\n            path: source.path || 'unknown'\n          };\n        } else if (source.source === 'collection') {\n          versions.collection = {\n            version: source.version || 'unknown',\n            lastModified: source.lastModified,\n            path: source.path || 'unknown'\n          };\n        }\n      }\n      \n      // Determine recommendation\n      const recommendation = this.determineVersionRecommendation(versions);\n      \n      return {\n        name: duplicate.name,\n        elementType: duplicate.elementType,\n        versions,\n        recommended: recommendation,\n        updateAvailable: recommendation.source !== 'local' && !!versions.local,\n        updateFrom: recommendation.source !== 'local' ? recommendation.source : undefined\n      };\n      \n    } catch (error) {\n      ErrorHandler.logError('UnifiedIndexManager.getVersionComparison', error, { name });\n      return null;\n    }\n  }\n\n  /**\n   * Get comprehensive statistics across all sources\n   */\n  public async getStats(): Promise<UnifiedIndexStats> {\n    try {\n      const [localStats, githubStats, collectionStats] = await Promise.allSettled([\n        this.getLocalStats(),\n        this.getGitHubStats(),\n        this.getCollectionStats()\n      ]);\n      \n      const local = localStats.status === 'fulfilled' ? localStats.value : {\n        totalElements: 0,\n        elementsByType: {} as Record<ElementType, number>,\n        lastBuilt: null,\n        isStale: true\n      };\n      \n      const github = githubStats.status === 'fulfilled' ? githubStats.value : {\n        totalElements: 0,\n        elementsByType: {} as Record<ElementType, number>,\n        lastFetched: null,\n        isStale: true\n      };\n      \n      const collection = collectionStats.status === 'fulfilled' ? collectionStats.value : {\n        totalElements: 0,\n        elementsByType: {} as Record<string, number>,\n        lastFetched: null,\n        isStale: true\n      };\n      \n      // Calculate combined statistics\n      const totalElements = local.totalElements + github.totalElements + collection.totalElements;\n      const duplicatesCount = await this.calculateDuplicatesCount();\n      const uniqueElements = totalElements - duplicatesCount;\n      \n      return {\n        local,\n        github,\n        collection,\n        combined: {\n          totalElements,\n          uniqueElements,\n          duplicates: duplicatesCount\n        },\n        performance: {\n          averageSearchTime: this.getPerformanceStats().searchStats.averageTime || 0,\n          cacheHitRate: this.getPerformanceStats().searchStats.cacheHitRate || 0,\n          lastOptimized: null\n        }\n      };\n      \n    } catch (error) {\n      ErrorHandler.logError('UnifiedIndexManager.getStats', error);\n      throw error;\n    }\n  }\n\n  /**\n   * Invalidate caches after user actions with performance monitoring\n   */\n  public invalidateAfterAction(action: string): void {\n    logger.info('Invalidating unified portfolio caches after user action', { action });\n    \n    // Clear result and index caches\n    this.resultCache.clear();\n    this.indexCache.clear();\n    \n    // Invalidate local cache\n    this.localIndexManager.rebuildIndex().catch(error => {\n      logger.warn('Failed to rebuild local index after action', {\n        action,\n        error: error instanceof Error ? error.message : String(error)\n      });\n    });\n    \n    // Invalidate GitHub cache\n    this.githubIndexer.invalidateAfterAction(action);\n    \n    // Invalidate collection cache\n    this.collectionIndexCache.clearCache().catch(error => {\n      logger.warn('Failed to clear collection cache after action', {\n        action,\n        error: error instanceof Error ? error.message : String(error)\n      });\n    });\n    \n    // Trigger garbage collection if memory usage is high\n    this.triggerMemoryCleanup();\n  }\n\n  /**\n   * Force rebuild of all indexes with performance optimization\n   */\n  public async rebuildAll(): Promise<void> {\n    const startTime = Date.now();\n    logger.info('Rebuilding all portfolio indexes with optimization...');\n    \n    try {\n      // Clear all caches\n      this.resultCache.clear();\n      this.indexCache.clear();\n      \n      // Reset performance counters\n      this.performanceMonitor.reset();\n      \n      // Rebuild in parallel with memory monitoring\n      const rebuildPromises = [\n        this.localIndexManager.rebuildIndex(),\n        this.githubIndexer.clearCache(),\n        this.collectionIndexCache.clearCache()\n      ];\n      \n      await Promise.all(rebuildPromises);\n      \n      // Trigger cleanup\n      this.triggerMemoryCleanup();\n      \n      const duration = Date.now() - startTime;\n      logger.info('All portfolio indexes rebuilt successfully', {\n        duration: `${duration}ms`,\n        memoryUsageMB: process.memoryUsage().heapUsed / (1024 * 1024)\n      });\n    } catch (error) {\n      ErrorHandler.logError('UnifiedIndexManager.rebuildAll', error);\n      throw error;\n    }\n  }\n\n  // =====================================================\n  // PRIVATE HELPER METHODS\n  // =====================================================\n  \n  /**\n   * Search with fallback strategies for resilient operation\n   */\n  private async searchWithFallback(source: 'local' | 'github' | 'collection', query: string, options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    const startTime = Date.now();\n    \n    try {\n      let results: UnifiedSearchResult[] = [];\n      \n      switch (source) {\n        case 'local':\n          results = await this.searchLocal(query, options);\n          break;\n        case 'github':\n          results = await this.searchGitHub(query, options);\n          break;\n        case 'collection':\n          results = await this.searchCollection(query, options);\n          break;\n      }\n      \n      logger.debug(`${source} search completed in ${Date.now() - startTime}ms with ${results.length} results`);\n      return results;\n      \n    } catch (error) {\n      logger.debug(`${source} search failed, attempting fallback`, {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      \n      // Fallback strategies\n      return await this.handleSearchFallback(source, query, options, error);\n    }\n  }\n  \n  /**\n   * Handle search fallback strategies\n   */\n  private async handleSearchFallback(source: 'local' | 'github' | 'collection', query: string, options: UnifiedSearchOptions, originalError: any): Promise<UnifiedSearchResult[]> {\n    try {\n      switch (source) {\n        case 'local':\n          // Try to use stale local index\n          logger.debug('Attempting to use stale local index');\n          return await this.searchLocalStale(query, options);\n          \n        case 'github':\n          // Try cached GitHub data\n          logger.debug('Attempting to use cached GitHub data');\n          return await this.searchGitHubCached(query, options);\n          \n        case 'collection':\n          // Try cached collection data\n          logger.debug('Attempting to use cached collection data');\n          return await this.searchCollectionCached(query, options);\n          \n        default:\n          return [];\n      }\n    } catch (fallbackError) {\n      logger.warn(`All fallback strategies failed for ${source}`, {\n        originalError: originalError instanceof Error ? originalError.message : String(originalError),\n        fallbackError: fallbackError instanceof Error ? fallbackError.message : String(fallbackError)\n      });\n      return [];\n    }\n  }\n  \n  /**\n   * Search local portfolio\n   */\n  private async searchLocal(query: string, options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    const localOptions = this.convertToLocalOptions(options);\n    const results = await this.localIndexManager.search(query, localOptions);\n    \n    return results.map(result => ({\n      source: 'local' as const,\n      entry: this.convertLocalEntry(result.entry),\n      matchType: result.matchType,\n      score: result.score,\n      version: result.entry.metadata.version\n    }));\n  }\n  \n  /**\n   * Search local with stale data fallback\n   */\n  private async searchLocalStale(query: string, options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    try {\n      // Try to get any local data, even stale\n      const localOptions = this.convertToLocalOptions(options);\n      const results = await this.localIndexManager.search(query, localOptions);\n      \n      return results.map(result => ({\n        source: 'local' as const,\n        entry: this.convertLocalEntry(result.entry),\n        matchType: result.matchType,\n        score: result.score * 0.8, // Reduce score for stale data\n        version: result.entry.metadata.version\n      }));\n    } catch {\n      return [];\n    }\n  }\n\n  /**\n   * Search GitHub portfolio\n   */\n  private async searchGitHub(query: string, options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    try {\n      const githubIndex = await this.githubIndexer.getIndex();\n      const results: UnifiedSearchResult[] = [];\n      \n      const queryLower = query.toLowerCase();\n      const queryTokens = queryLower.split(/\\s+/).filter(token => token.length > 0);\n      \n      if (queryTokens.length === 0 && query.trim() !== '') {\n        return results;\n      }\n      \n      // Search across all GitHub elements\n      for (const [elementType, entries] of githubIndex.elements) {\n        // Filter by element type if specified\n        if (options.elementType && elementType !== options.elementType) {\n          continue;\n        }\n        \n        for (const entry of entries) {\n          const score = this.calculateGitHubMatchScore(entry, queryTokens, query);\n          if (score > 0 || query.trim() === '') {\n            results.push({\n              source: 'github' as const,\n              entry: this.convertGitHubEntry(entry),\n              matchType: this.determineMatchType(entry, queryTokens),\n              score: query.trim() === '' ? 1 : score, // Default score for empty query\n              version: entry.version\n            });\n          }\n        }\n      }\n      \n      return results.sort((a, b) => b.score - a.score);\n      \n    } catch (error) {\n      logger.debug('GitHub search failed', {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      throw error; // Re-throw to trigger fallback\n    }\n  }\n  \n  /**\n   * Search GitHub with cached data fallback\n   */\n  private async searchGitHubCached(query: string, options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    try {\n      // Try to use stale GitHub data\n      const cacheStats = this.githubIndexer.getCacheStats();\n      if (!cacheStats.isStale) {\n        return await this.searchGitHub(query, options);\n      }\n      \n      // Use stale data with reduced scores\n      const results = await this.searchGitHub(query, options);\n      return results.map(result => ({\n        ...result,\n        score: result.score * 0.7 // Reduce score for stale data\n      }));\n      \n    } catch {\n      return [];\n    }\n  }\n  \n  /**\n   * Search collection portfolio\n   */\n  private async searchCollection(query: string, options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    try {\n      const collectionIndex = await this.collectionIndexCache.getIndex();\n      const results: UnifiedSearchResult[] = [];\n      \n      const queryLower = query.toLowerCase();\n      const queryTokens = queryLower.split(/\\s+/).filter(token => token.length > 0);\n      \n      if (queryTokens.length === 0 && query.trim() !== '') {\n        return results;\n      }\n      \n      // Search across all collection elements\n      for (const [elementType, entries] of Object.entries(collectionIndex.index)) {\n        // Filter by element type if specified\n        if (options.elementType && elementType !== options.elementType.toString()) {\n          continue;\n        }\n        \n        for (const entry of entries) {\n          const score = this.calculateCollectionMatchScore(entry, queryTokens, query);\n          if (score > 0 || query.trim() === '') {\n            results.push({\n              source: 'collection' as const,\n              entry: this.convertCollectionEntry(entry, elementType),\n              matchType: this.determineCollectionMatchType(entry, queryTokens),\n              score: query.trim() === '' ? 1 : score, // Default score for empty query\n              version: entry.version\n            });\n          }\n        }\n      }\n      \n      return results.sort((a, b) => b.score - a.score);\n      \n    } catch (error) {\n      logger.debug('Collection search failed', {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      throw error; // Re-throw to trigger fallback\n    }\n  }\n  \n  /**\n   * Search collection with cached data fallback\n   */\n  private async searchCollectionCached(query: string, options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    try {\n      // Try to use stale collection data\n      const cacheStats = this.collectionIndexCache.getCacheStats();\n      if (cacheStats.isValid) {\n        return await this.searchCollection(query, options);\n      }\n      \n      // Use stale data with reduced scores\n      const results = await this.searchCollection(query, options);\n      return results.map(result => ({\n        ...result,\n        score: result.score * 0.6 // Reduce score for stale collection data\n      }));\n      \n    } catch {\n      return [];\n    }\n  }\n\n  /**\n   * Process search results with advanced features\n   */\n  private async processSearchResults(results: UnifiedSearchResult[], options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    // Apply smart ranking\n    const rankedResults = this.applySmartRanking(results, options);\n    \n    // Detect duplicates and version conflicts\n    const processedResults = await this.detectDuplicatesAndConflicts(rankedResults);\n    \n    // Apply sorting\n    const sortedResults = this.applySorting(processedResults, options.sortBy || 'relevance', options.query);\n    \n    return sortedResults;\n  }\n  \n  /**\n   * Apply smart result ranking\n   */\n  private applySmartRanking(results: UnifiedSearchResult[], options: UnifiedSearchOptions): UnifiedSearchResult[] {\n    return results.map(result => {\n      let adjustedScore = result.score;\n      \n      // No location-based scoring - score should be based on relevance only\n      // Source location doesn't affect the intrinsic value of an element\n      \n      // Consider version freshness (newer versions get small bonus)\n      if (result.version && result.version !== 'unknown') {\n        const versionParts = result.version.split('.');\n        if (versionParts.length >= 2) {\n          const major = parseInt(versionParts[0]) || 0;\n          const minor = parseInt(versionParts[1]) || 0;\n          adjustedScore += (major * 0.1) + (minor * 0.01);\n        }\n      }\n      \n      // Boost exact matches\n      if (result.entry.name.toLowerCase() === options.query.toLowerCase()) {\n        adjustedScore *= 2.0;\n      }\n      \n      return {\n        ...result,\n        score: adjustedScore\n      };\n    });\n  }\n  \n  /**\n   * Detect duplicates and version conflicts\n   */\n  private async detectDuplicatesAndConflicts(results: UnifiedSearchResult[]): Promise<UnifiedSearchResult[]> {\n    const nameMap = new Map<string, UnifiedSearchResult[]>();\n    \n    // Group by name and element type\n    for (const result of results) {\n      const key = `${result.entry.elementType}:${result.entry.name.toLowerCase()}`;\n      if (!nameMap.has(key)) {\n        nameMap.set(key, []);\n      }\n      nameMap.get(key)!.push(result);\n    }\n    \n    const processedResults: UnifiedSearchResult[] = [];\n    \n    // Process each group\n    for (const [key, groupResults] of nameMap) {\n      if (groupResults.length === 1) {\n        // No duplicates\n        processedResults.push(groupResults[0]);\n      } else {\n        // Has duplicates - detect version conflicts\n        const versionConflict = this.detectVersionConflictFromResults(groupResults);\n        \n        // Mark all results as duplicates and add conflict info\n        for (const result of groupResults) {\n          processedResults.push({\n            ...result,\n            isDuplicate: true,\n            versionConflict\n          });\n        }\n      }\n    }\n    \n    return processedResults;\n  }\n  \n  /**\n   * Apply pagination to results\n   */\n  private applyPagination(results: UnifiedSearchResult[], options: UnifiedSearchOptions): UnifiedSearchResult[] {\n    const page = options.page || 1;\n    const pageSize = options.pageSize || 20;\n    const startIndex = (page - 1) * pageSize;\n    const endIndex = startIndex + pageSize;\n    \n    return results.slice(startIndex, endIndex);\n  }\n  \n  /**\n   * Apply sorting to results\n   */\n  private applySorting(results: UnifiedSearchResult[], sortBy: 'relevance' | 'source' | 'name' | 'version', query: string): UnifiedSearchResult[] {\n    const sorted = [...results];\n    \n    switch (sortBy) {\n      case 'name':\n        sorted.sort((a, b) => a.entry.name.localeCompare(b.entry.name));\n        break;\n      case 'source':\n        sorted.sort((a, b) => {\n          const sourceOrder = { 'local': 0, 'github': 1, 'collection': 2 };\n          return sourceOrder[a.source] - sourceOrder[b.source];\n        });\n        break;\n      case 'version':\n        sorted.sort((a, b) => this.compareVersions(b.version || '0', a.version || '0'));\n        break;\n      case 'relevance':\n      default:\n        sorted.sort((a, b) => b.score - a.score);\n        break;\n    }\n    \n    return sorted;\n  }\n  \n  /**\n   * Calculate match score for GitHub entries\n   */\n  private calculateGitHubMatchScore(entry: GitHubIndexEntry, queryTokens: string[], query: string): number {\n    if (queryTokens.length === 0) return 1; // Default score for empty query\n    \n    let score = 0;\n    \n    const name = entry.name.toLowerCase();\n    const description = (entry.description || '').toLowerCase();\n    const path = (entry.path || '').toLowerCase();\n    \n    // Check name matches\n    for (const token of queryTokens) {\n      if (name.includes(token)) {\n        score += name === token ? 10 : (name.startsWith(token) ? 5 : 2);\n      }\n      if (description.includes(token)) {\n        score += 3;\n      }\n      if (path.includes(token)) {\n        score += 1;\n      }\n    }\n    \n    // Exact query match bonus\n    if (name.includes(query.toLowerCase())) {\n      score += query.length > 3 ? 15 : 10;\n    }\n    \n    return score;\n  }\n  \n  /**\n   * Calculate match score for collection entries\n   */\n  private calculateCollectionMatchScore(entry: CollectionIndexEntry, queryTokens: string[], query: string): number {\n    if (queryTokens.length === 0) return 1; // Default score for empty query\n    \n    let score = 0;\n    \n    const name = entry.name.toLowerCase();\n    const description = (entry.description || '').toLowerCase();\n    const path = (entry.path || '').toLowerCase();\n    const tags = entry.tags.map(tag => tag.toLowerCase()).join(' ');\n    \n    // Check matches across all fields\n    for (const token of queryTokens) {\n      if (name.includes(token)) {\n        score += name === token ? 10 : (name.startsWith(token) ? 5 : 2);\n      }\n      if (description.includes(token)) {\n        score += 3;\n      }\n      if (path.includes(token)) {\n        score += 1;\n      }\n      if (tags.includes(token)) {\n        score += 4;\n      }\n    }\n    \n    // Exact query match bonus\n    if (name.includes(query.toLowerCase())) {\n      score += query.length > 3 ? 15 : 10;\n    }\n    \n    return score;\n  }\n\n  /**\n   * Get all elements by type across sources\n   */\n  private async getAllElementsByType(elementType: ElementType, options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    const promises: Promise<UnifiedSearchResult[]>[] = [];\n    \n    if (options.includeLocal) {\n      promises.push(this.getLocalElementsByType(elementType));\n    }\n    if (options.includeGitHub) {\n      promises.push(this.getGitHubElementsByType(elementType));\n    }\n    if (options.includeCollection) {\n      promises.push(this.getCollectionElementsByType(elementType));\n    }\n    \n    const results = await Promise.allSettled(promises);\n    const allResults: UnifiedSearchResult[] = [];\n    \n    results.forEach(result => {\n      if (result.status === 'fulfilled') {\n        allResults.push(...result.value);\n      }\n    });\n    \n    return allResults;\n  }\n  \n  /**\n   * Get local elements by type\n   */\n  private async getLocalElementsByType(elementType: ElementType): Promise<UnifiedSearchResult[]> {\n    try {\n      const elements = await this.localIndexManager.getElementsByType(elementType);\n      return elements.map(entry => ({\n        source: 'local' as const,\n        entry: this.convertLocalEntry(entry),\n        matchType: 'type',\n        score: 1,\n        version: entry.metadata.version\n      }));\n    } catch {\n      return [];\n    }\n  }\n  \n  /**\n   * Get GitHub elements by type\n   */\n  private async getGitHubElementsByType(elementType: ElementType): Promise<UnifiedSearchResult[]> {\n    try {\n      const githubIndex = await this.githubIndexer.getIndex();\n      const entries = githubIndex.elements.get(elementType) || [];\n      \n      return entries.map(entry => ({\n        source: 'github' as const,\n        entry: this.convertGitHubEntry(entry),\n        matchType: 'type',\n        score: 1,\n        version: entry.version\n      }));\n    } catch {\n      return [];\n    }\n  }\n  \n  /**\n   * Get collection elements by type\n   */\n  private async getCollectionElementsByType(elementType: ElementType): Promise<UnifiedSearchResult[]> {\n    try {\n      const collectionIndex = await this.collectionIndexCache.getIndex();\n      const entries = collectionIndex.index[elementType.toString()] || [];\n      \n      return entries.map(entry => ({\n        source: 'collection' as const,\n        entry: this.convertCollectionEntry(entry, elementType.toString()),\n        matchType: 'type',\n        score: 1,\n        version: entry.version\n      }));\n    } catch {\n      return [];\n    }\n  }\n\n  /**\n   * Get local portfolio statistics\n   */\n  private async getLocalStats(): Promise<UnifiedIndexStats['local']> {\n    return await this.localIndexManager.getStats();\n  }\n  \n  /**\n   * Get GitHub portfolio statistics\n   */\n  private async getGitHubStats(): Promise<UnifiedIndexStats['github']> {\n    const cacheStats = this.githubIndexer.getCacheStats();\n    const githubIndex = await this.githubIndexer.getIndex();\n    \n    const elementsByType: Record<ElementType, number> = {} as Record<ElementType, number>;\n    for (const elementType of Object.values(ElementType)) {\n      elementsByType[elementType] = (githubIndex.elements.get(elementType) || []).length;\n    }\n    \n    return {\n      totalElements: githubIndex.totalElements,\n      elementsByType,\n      lastFetched: cacheStats.lastFetch,\n      isStale: cacheStats.isStale,\n      username: githubIndex.username,\n      repository: githubIndex.repository\n    };\n  }\n  \n  /**\n   * Get collection portfolio statistics\n   */\n  private async getCollectionStats(): Promise<UnifiedIndexStats['collection']> {\n    const cacheStats = this.collectionIndexCache.getCacheStats();\n    const collectionIndex = await this.collectionIndexCache.getIndex();\n    \n    const elementsByType: Record<string, number> = {};\n    for (const [elementType, entries] of Object.entries(collectionIndex.index)) {\n      elementsByType[elementType] = entries.length;\n    }\n    \n    return {\n      totalElements: collectionIndex.total_elements,\n      elementsByType,\n      lastFetched: cacheStats.hasCache ? new Date(Date.now() - cacheStats.age) : null,\n      isStale: !cacheStats.isValid,\n      version: collectionIndex.version\n    };\n  }\n  \n  /**\n   * Calculate duplicates count across all sources\n   */\n  private async calculateDuplicatesCount(): Promise<number> {\n    try {\n      // This is a placeholder - actual implementation would need optimization\n      // For now, return 0 to avoid the expensive operation during stats calculation\n      return 0;\n    } catch {\n      return 0;\n    }\n  }\n\n  /**\n   * Convert local index entry to unified format\n   */\n  private convertLocalEntry(entry: IndexEntry): UnifiedIndexEntry {\n    return {\n      name: entry.metadata.name,\n      description: entry.metadata.description,\n      version: entry.metadata.version,\n      author: entry.metadata.author,\n      elementType: entry.elementType,\n      lastModified: entry.lastModified,\n      source: 'local',\n      localFilePath: entry.filePath,\n      filename: entry.filename,\n      tags: entry.metadata.tags,\n      keywords: entry.metadata.keywords,\n      triggers: entry.metadata.triggers,\n      category: entry.metadata.category\n    };\n  }\n\n  /**\n   * Convert GitHub index entry to unified format\n   */\n  private convertGitHubEntry(entry: GitHubIndexEntry): UnifiedIndexEntry {\n    return {\n      name: entry.name,\n      description: entry.description,\n      version: entry.version,\n      author: entry.author,\n      elementType: entry.elementType,\n      lastModified: entry.lastModified,\n      source: 'github',\n      githubPath: entry.path,\n      githubSha: entry.sha,\n      githubHtmlUrl: entry.htmlUrl,\n      githubDownloadUrl: entry.downloadUrl,\n      githubSize: entry.size\n    };\n  }\n  \n  /**\n   * Convert collection index entry to unified format\n   */\n  private convertCollectionEntry(entry: CollectionIndexEntry, elementType: string): UnifiedIndexEntry {\n    return {\n      name: entry.name,\n      description: entry.description,\n      version: entry.version,\n      author: entry.author,\n      elementType: this.mapStringToElementType(elementType),\n      lastModified: new Date(entry.created),\n      source: 'collection',\n      collectionPath: entry.path,\n      collectionSha: entry.sha,\n      collectionTags: entry.tags,\n      collectionCategory: entry.category,\n      collectionLicense: entry.license\n    };\n  }\n  \n  /**\n   * Map string to ElementType enum\n   */\n  private mapStringToElementType(elementType: string): ElementType {\n    // Handle mapping from collection element types to our ElementType enum\n    switch (elementType.toLowerCase()) {\n      case 'personas':\n        return ElementType.PERSONA;\n      case 'skills':\n        return ElementType.SKILL;\n      case 'agents':\n        return ElementType.AGENT;\n      case 'prompts':\n      case 'templates':\n        return ElementType.TEMPLATE; // Map prompts and templates to TEMPLATE\n      case 'tools':\n        return ElementType.SKILL; // Map tools to SKILL as fallback\n      case 'ensembles':\n        return ElementType.ENSEMBLE;\n      case 'memories':\n        return ElementType.MEMORY;\n      default:\n        return ElementType.SKILL; // Default fallback\n    }\n  }\n  \n  /**\n   * Convert unified search options to local search options\n   */\n  private convertToLocalOptions(options: UnifiedSearchOptions): SearchOptions {\n    return {\n      elementType: options.elementType,\n      maxResults: options.pageSize || 20\n    };\n  }\n  \n  /**\n   * Determine match type for GitHub entries\n   */\n  private determineMatchType(entry: GitHubIndexEntry, queryTokens: string[]): string {\n    const name = entry.name.toLowerCase();\n    const description = (entry.description || '').toLowerCase();\n    \n    // Check what matched\n    for (const token of queryTokens) {\n      if (name.includes(token)) {\n        return name === token ? 'exact_name' : 'name';\n      }\n      if (description.includes(token)) {\n        return 'description';\n      }\n    }\n    \n    return 'content';\n  }\n  \n  /**\n   * Determine match type for collection entries\n   */\n  private determineCollectionMatchType(entry: CollectionIndexEntry, queryTokens: string[]): string {\n    const name = entry.name.toLowerCase();\n    const description = (entry.description || '').toLowerCase();\n    const tags = entry.tags.map(tag => tag.toLowerCase()).join(' ');\n    \n    // Check what matched\n    for (const token of queryTokens) {\n      if (name.includes(token)) {\n        return name === token ? 'exact_name' : 'name';\n      }\n      if (description.includes(token)) {\n        return 'description';\n      }\n      if (tags.includes(token)) {\n        return 'tag';\n      }\n    }\n    \n    return 'content';\n  }\n  \n  /**\n   * Get path from unified entry\n   */\n  private getPathFromEntry(entry: UnifiedIndexEntry): string {\n    switch (entry.source) {\n      case 'local':\n        return entry.localFilePath || entry.filename || 'unknown';\n      case 'github':\n        return entry.githubPath || 'unknown';\n      case 'collection':\n        return entry.collectionPath || 'unknown';\n      default:\n        return 'unknown';\n    }\n  }\n  \n  /**\n   * Detect version conflict from sources\n   */\n  private detectVersionConflict(sources: DuplicateInfo['sources']): VersionConflict | undefined {\n    const versions = new Map<string, 'local' | 'github' | 'collection'>();\n    \n    for (const source of sources) {\n      if (source.version && source.version !== 'unknown') {\n        versions.set(source.version, source.source);\n      }\n    }\n    \n    if (versions.size <= 1) {\n      return undefined; // No conflict if all versions are the same or missing\n    }\n    \n    // Build version conflict info\n    const versionConflict: VersionConflict = {\n      recommended: 'local',\n      reason: 'Multiple versions detected'\n    };\n    \n    for (const source of sources) {\n      if (source.version) {\n        versionConflict[source.source] = source.version;\n      }\n    }\n    \n    // Determine recommendation\n    const recommendation = this.determineVersionRecommendationFromSources(sources);\n    versionConflict.recommended = recommendation.source;\n    versionConflict.reason = recommendation.reason;\n    \n    return versionConflict;\n  }\n  \n  /**\n   * Detect version conflict from search results\n   */\n  private detectVersionConflictFromResults(results: UnifiedSearchResult[]): VersionConflict | undefined {\n    const sources = results.map(result => ({\n      source: result.source,\n      version: result.version,\n      lastModified: result.entry.lastModified,\n      path: this.getPathFromEntry(result.entry)\n    }));\n    \n    return this.detectVersionConflict(sources);\n  }\n  \n  /**\n   * Determine version recommendation from version info\n   */\n  private determineVersionRecommendation(versions: VersionInfo['versions']): { source: 'local' | 'github' | 'collection'; reason: string } {\n    // Prefer local if available and not too old\n    if (versions.local) {\n      const localAge = Date.now() - versions.local.lastModified.getTime();\n      const sevenDays = 7 * 24 * 60 * 60 * 1000;\n      \n      if (localAge < sevenDays) {\n        return { source: 'local', reason: 'Local version is recent and authoritative' };\n      }\n    }\n    \n    // Compare versions if available\n    const versionEntries = Object.entries(versions).filter(([_, info]) => info?.version);\n    \n    if (versionEntries.length > 1) {\n      // Find highest version\n      let highest: { source: 'local' | 'github' | 'collection'; version: string } = {\n        source: 'local',\n        version: '0.0.0'\n      };\n      \n      for (const [source, info] of versionEntries) {\n        if (info && this.compareVersions(info.version, highest.version) > 0) {\n          highest = {\n            source: source as 'local' | 'github' | 'collection',\n            version: info.version\n          };\n        }\n      }\n      \n      return { source: highest.source, reason: `Highest version (${highest.version})` };\n    }\n    \n    // Fallback to most recent\n    let mostRecent: { source: 'local' | 'github' | 'collection'; date: Date } = {\n      source: 'local',\n      date: new Date(0)\n    };\n    \n    for (const [source, info] of Object.entries(versions)) {\n      if (info && info.lastModified > mostRecent.date) {\n        mostRecent = {\n          source: source as 'local' | 'github' | 'collection',\n          date: info.lastModified\n        };\n      }\n    }\n    \n    return { source: mostRecent.source, reason: 'Most recently modified' };\n  }\n  \n  /**\n   * Determine version recommendation from sources\n   */\n  private determineVersionRecommendationFromSources(sources: DuplicateInfo['sources']): { source: 'local' | 'github' | 'collection'; reason: string } {\n    // Convert sources to versions format\n    const versions: VersionInfo['versions'] = {};\n    \n    for (const source of sources) {\n      if (source.source === 'local') {\n        versions.local = {\n          version: source.version || 'unknown',\n          lastModified: source.lastModified,\n          path: source.path || 'unknown'\n        };\n      } else if (source.source === 'github') {\n        versions.github = {\n          version: source.version || 'unknown',\n          lastModified: source.lastModified,\n          path: source.path || 'unknown'\n        };\n      } else if (source.source === 'collection') {\n        versions.collection = {\n          version: source.version || 'unknown',\n          lastModified: source.lastModified,\n          path: source.path || 'unknown'\n        };\n      }\n    }\n    \n    return this.determineVersionRecommendation(versions);\n  }\n  \n  /**\n   * Compare semantic versions\n   */\n  private compareVersions(a: string, b: string): number {\n    const parseVersion = (version: string) => {\n      const parts = version.split('.').map(part => parseInt(part) || 0);\n      return [parts[0] || 0, parts[1] || 0, parts[2] || 0];\n    };\n    \n    const [aMajor, aMinor, aPatch] = parseVersion(a);\n    const [bMajor, bMinor, bPatch] = parseVersion(b);\n    \n    if (aMajor !== bMajor) return aMajor - bMajor;\n    if (aMinor !== bMinor) return aMinor - bMinor;\n    return aPatch - bPatch;\n  }\n\n  /**\n   * Remove duplicate results based on name and type\n   */\n  private deduplicateResults(results: UnifiedSearchResult[]): UnifiedSearchResult[] {\n    const seen = new Set<string>();\n    const deduplicated: UnifiedSearchResult[] = [];\n    \n    for (const result of results) {\n      const key = `${result.entry.elementType}:${result.entry.name.toLowerCase()}`;\n      if (!seen.has(key)) {\n        seen.add(key);\n        deduplicated.push(result);\n      }\n    }\n    \n    return deduplicated;\n  }\n\n  /**\n   * Remove duplicate entries based on name and type\n   */\n  private deduplicateEntries(entries: UnifiedIndexEntry[]): UnifiedIndexEntry[] {\n    const seen = new Set<string>();\n    const deduplicated: UnifiedIndexEntry[] = [];\n    \n    for (const entry of entries) {\n      const key = `${entry.elementType}:${entry.name.toLowerCase()}`;\n      if (!seen.has(key)) {\n        seen.add(key);\n        deduplicated.push(entry);\n      }\n    }\n    \n    return deduplicated;\n  }\n  \n  // =====================================================\n  // PERFORMANCE MONITORING AND OPTIMIZATION\n  // =====================================================\n  \n  /**\n   * Stream search results for better performance with large datasets\n   */\n  private async streamSearch(options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    const { query, cursor, maxResults = 1000 } = options;\n    const startTime = Date.now();\n    \n    logger.debug('Starting streaming search', { query: query.substring(0, 50), cursor, maxResults });\n    \n    const results: UnifiedSearchResult[] = [];\n    const sources = this.getEnabledSources(options);\n    \n    // Process sources in sequence for memory efficiency\n    for (const source of sources) {\n      if (results.length >= maxResults) {\n        break;\n      }\n      \n      try {\n        const sourceResults = await this.searchWithFallback(source, query, {\n          ...options,\n          pageSize: Math.min(this.BATCH_SIZE, maxResults - results.length)\n        });\n        \n        results.push(...sourceResults);\n        \n        // Add cursor information for pagination\n        sourceResults.forEach((result, index) => {\n          result.cursor = this.generateCursor(source, index);\n        });\n        \n      } catch (error) {\n        logger.warn(`Streaming search failed for source ${source}`, {\n          error: error instanceof Error ? error.message : String(error)\n        });\n      }\n    }\n    \n    logger.debug('Streaming search completed', {\n      resultCount: results.length,\n      duration: `${Date.now() - startTime}ms`\n    });\n    \n    return results;\n  }\n  \n  /**\n   * Process search results with memory-efficient batching\n   */\n  private async processSearchResultsOptimized(results: UnifiedSearchResult[], options: UnifiedSearchOptions): Promise<UnifiedSearchResult[]> {\n    if (results.length === 0) {\n      return results;\n    }\n    \n    // Process in batches to avoid memory spikes with large result sets\n    const batchSize = Math.min(this.BATCH_SIZE, results.length);\n    const processedResults: UnifiedSearchResult[] = [];\n    \n    for (let i = 0; i < results.length; i += batchSize) {\n      const batch = results.slice(i, i + batchSize);\n      \n      // Apply smart ranking\n      const rankedBatch = this.applySmartRanking(batch, options);\n      \n      // Detect duplicates and conflicts\n      const processedBatch = await this.detectDuplicatesAndConflicts(rankedBatch);\n      \n      processedResults.push(...processedBatch);\n      \n      // Yield control to prevent blocking\n      if (i % (batchSize * 4) === 0) {\n        await new Promise(resolve => setImmediate(resolve));\n      }\n    }\n    \n    // Apply final sorting\n    return this.applySorting(processedResults, options.sortBy || 'relevance', options.query);\n  }\n  \n  /**\n   * Get enabled search sources\n   */\n  private getEnabledSources(options: UnifiedSearchOptions): ('local' | 'github' | 'collection')[] {\n    const sources: ('local' | 'github' | 'collection')[] = [];\n    \n    if (options.includeLocal !== false) sources.push('local');\n    if (options.includeGitHub !== false) sources.push('github');\n    if (options.includeCollection === true) sources.push('collection');\n    \n    return sources;\n  }\n  \n  /**\n   * Batch sources for concurrent processing\n   */\n  private batchSources(sources: ('local' | 'github' | 'collection')[], batchSize: number): ('local' | 'github' | 'collection')[][] {\n    const batches: ('local' | 'github' | 'collection')[][] = [];\n    \n    for (let i = 0; i < sources.length; i += batchSize) {\n      batches.push(sources.slice(i, i + batchSize));\n    }\n    \n    return batches;\n  }\n  \n  /**\n   * Generate cursor for pagination\n   */\n  private generateCursor(source: string, index: number): string {\n    const timestamp = Date.now();\n    return Buffer.from(`${source}:${index}:${timestamp}`).toString('base64');\n  }\n  \n  /**\n   * Trigger memory cleanup when usage is high\n   */\n  private triggerMemoryCleanup(): void {\n    // Force cache cleanup\n    this.resultCache.cleanup();\n    this.indexCache.cleanup();\n    \n    // Suggest garbage collection\n    if (global.gc) {\n      global.gc();\n      logger.debug('Triggered garbage collection');\n    }\n  }\n  \n  /**\n   * Record search performance metrics\n   */\n  private recordSearchMetrics(metrics: SearchMetrics): void {\n    this.performanceMonitor.recordSearch(metrics);\n    \n    // Update cache performance metrics\n    const cacheStats = this.resultCache.getStats();\n    this.performanceMonitor.recordCachePerformance('searchResults', {\n      hitRate: cacheStats.hitRate,\n      avgHitTime: 1, // Placeholder\n      avgMissTime: 5, // Placeholder\n      totalHits: cacheStats.hitCount,\n      totalMisses: cacheStats.missCount,\n      evictions: cacheStats.evictionCount\n    });\n  }\n  \n  /**\n   * Create cache key for search options\n   */\n  private createCacheKey(options: UnifiedSearchOptions): string {\n    return JSON.stringify({\n      query: options.query,\n      includeLocal: options.includeLocal,\n      includeGitHub: options.includeGitHub,\n      includeCollection: options.includeCollection,\n      elementType: options.elementType,\n      page: options.page,\n      pageSize: options.pageSize,\n      sortBy: options.sortBy,\n      lazyLoad: options.lazyLoad\n    });\n  }\n  \n  /**\n   * Get performance statistics\n   */\n  public getPerformanceStats(): {\n    searchStats: any;\n    memoryStats: any;\n    cacheStats: any;\n    trends: any;\n  } {\n    return {\n      searchStats: this.performanceMonitor.getSearchStats(),\n      memoryStats: this.performanceMonitor.getMemoryStats(),\n      cacheStats: {\n        searchResults: this.resultCache.getStats(),\n        indexCache: this.indexCache.getStats()\n      },\n      trends: this.performanceMonitor.analyzeTrends()\n    };\n  }\n}"]}