@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.
- package/CHANGELOG.md +56 -0
- package/README.md +494 -111
- package/data/agents/code-reviewer.md +8 -1
- package/data/agents/research-assistant.md +8 -1
- package/data/agents/task-manager.md +8 -1
- package/data/ensembles/business-advisor.md +8 -1
- package/data/ensembles/creative-studio.md +8 -1
- package/data/ensembles/development-team.md +8 -1
- package/data/ensembles/security-analysis-team.md +8 -1
- package/data/memories/conversation-history.md +8 -1
- package/data/memories/learning-progress.md +8 -1
- package/data/memories/project-context.md +8 -1
- package/data/personas/business-consultant.md +8 -1
- package/data/personas/creative-writer.md +8 -1
- package/data/personas/debug-detective.md +8 -1
- package/data/personas/eli5-explainer.md +8 -1
- package/data/personas/security-analyst.md +8 -1
- package/data/personas/technical-analyst.md +8 -1
- package/data/skills/code-review.md +8 -1
- package/data/skills/creative-writing.md +8 -1
- package/data/skills/data-analysis.md +8 -1
- package/data/skills/penetration-testing.md +8 -1
- package/data/skills/research.md +8 -1
- package/data/skills/threat-modeling.md +8 -1
- package/data/skills/translation.md +8 -1
- package/data/templates/code-documentation.md +8 -1
- package/data/templates/email-professional.md +8 -1
- package/data/templates/meeting-notes.md +8 -1
- package/data/templates/penetration-test-report.md +8 -1
- package/data/templates/project-brief.md +8 -1
- package/data/templates/report-executive.md +8 -1
- package/data/templates/security-vulnerability-report.md +8 -1
- package/data/templates/threat-assessment-report.md +8 -1
- package/dist/auth/GitHubAuthManager.d.ts +6 -1
- package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
- package/dist/auth/GitHubAuthManager.js +45 -18
- package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
- package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
- package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
- package/dist/cache/CollectionCache.d.ts.map +1 -1
- package/dist/cache/CollectionCache.js +13 -3
- package/dist/cache/CollectionIndexCache.d.ts +77 -0
- package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
- package/dist/cache/CollectionIndexCache.js +349 -0
- package/dist/cache/LRUCache.d.ts +93 -0
- package/dist/cache/LRUCache.d.ts.map +1 -0
- package/dist/cache/LRUCache.js +299 -0
- package/dist/cache/index.d.ts +1 -0
- package/dist/cache/index.d.ts.map +1 -1
- package/dist/cache/index.js +2 -1
- package/dist/collection/CollectionBrowser.d.ts +21 -1
- package/dist/collection/CollectionBrowser.d.ts.map +1 -1
- package/dist/collection/CollectionBrowser.js +130 -10
- package/dist/collection/CollectionIndexManager.d.ts +151 -0
- package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
- package/dist/collection/CollectionIndexManager.js +499 -0
- package/dist/collection/CollectionSearch.d.ts +55 -0
- package/dist/collection/CollectionSearch.d.ts.map +1 -1
- package/dist/collection/CollectionSearch.js +338 -13
- package/dist/collection/CollectionSeeder.d.ts.map +1 -1
- package/dist/collection/CollectionSeeder.js +38 -1
- package/dist/collection/ElementInstaller.d.ts +31 -0
- package/dist/collection/ElementInstaller.d.ts.map +1 -1
- package/dist/collection/ElementInstaller.js +77 -15
- package/dist/collection/PersonaSubmitter.d.ts +1 -1
- package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
- package/dist/collection/PersonaSubmitter.js +2 -2
- package/dist/collection/index.d.ts +1 -0
- package/dist/collection/index.d.ts.map +1 -1
- package/dist/collection/index.js +2 -1
- package/dist/config/ConfigManager.d.ts +78 -0
- package/dist/config/ConfigManager.d.ts.map +1 -0
- package/dist/config/ConfigManager.js +216 -0
- package/dist/config/element-types.d.ts +135 -0
- package/dist/config/element-types.d.ts.map +1 -0
- package/dist/config/element-types.js +108 -0
- package/dist/config/index.d.ts +2 -0
- package/dist/config/index.d.ts.map +1 -1
- package/dist/config/index.js +3 -1
- package/dist/config/portfolio-constants.d.ts +83 -0
- package/dist/config/portfolio-constants.d.ts.map +1 -0
- package/dist/config/portfolio-constants.js +99 -0
- package/dist/elements/BaseElement.d.ts +14 -2
- package/dist/elements/BaseElement.d.ts.map +1 -1
- package/dist/elements/BaseElement.js +88 -6
- package/dist/elements/agents/Agent.d.ts +10 -1
- package/dist/elements/agents/Agent.d.ts.map +1 -1
- package/dist/elements/agents/Agent.js +66 -19
- package/dist/elements/agents/AgentManager.d.ts +2 -0
- package/dist/elements/agents/AgentManager.d.ts.map +1 -1
- package/dist/elements/agents/AgentManager.js +12 -10
- package/dist/elements/skills/Skill.d.ts +10 -1
- package/dist/elements/skills/Skill.d.ts.map +1 -1
- package/dist/elements/skills/Skill.js +40 -3
- package/dist/elements/skills/SkillManager.d.ts +1 -0
- package/dist/elements/skills/SkillManager.d.ts.map +1 -1
- package/dist/elements/skills/SkillManager.js +10 -4
- package/dist/elements/templates/Template.d.ts +10 -1
- package/dist/elements/templates/Template.d.ts.map +1 -1
- package/dist/elements/templates/Template.js +35 -18
- package/dist/elements/templates/TemplateManager.d.ts +1 -1
- package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
- package/dist/elements/templates/TemplateManager.js +6 -5
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/index.barrel.d.ts +1 -2
- package/dist/index.barrel.d.ts.map +1 -1
- package/dist/index.barrel.js +2 -4
- package/dist/index.d.ts +143 -25
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1883 -310
- package/dist/persona/PersonaElement.d.ts +10 -0
- package/dist/persona/PersonaElement.d.ts.map +1 -1
- package/dist/persona/PersonaElement.js +55 -32
- package/dist/persona/PersonaElementManager.d.ts.map +1 -1
- package/dist/persona/PersonaElementManager.js +13 -11
- package/dist/persona/PersonaLoader.d.ts.map +1 -1
- package/dist/persona/PersonaLoader.js +8 -2
- package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
- package/dist/persona/export-import/PersonaImporter.js +24 -5
- package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
- package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
- package/dist/persona/export-import/PersonaSharer.js +198 -22
- package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
- package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
- package/dist/portfolio/DefaultElementProvider.js +499 -7
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
- package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
- package/dist/portfolio/MigrationManager.d.ts.map +1 -1
- package/dist/portfolio/MigrationManager.js +136 -3
- package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
- package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioIndexManager.js +478 -0
- package/dist/portfolio/PortfolioManager.d.ts +5 -0
- package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioManager.js +61 -20
- package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
- package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioRepoManager.js +337 -0
- package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
- package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
- package/dist/portfolio/UnifiedIndexManager.js +1434 -0
- package/dist/portfolio/index.d.ts +15 -0
- package/dist/portfolio/index.d.ts.map +1 -0
- package/dist/portfolio/index.js +15 -0
- package/dist/portfolio/types.d.ts +7 -0
- package/dist/portfolio/types.d.ts.map +1 -1
- package/dist/portfolio/types.js +6 -1
- package/dist/security/InputValidator.d.ts.map +1 -1
- package/dist/security/InputValidator.js +50 -48
- package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
- package/dist/security/audit/SecurityAuditor.js +17 -9
- package/dist/security/audit/config/suppressions.d.ts.map +1 -1
- package/dist/security/audit/config/suppressions.js +19 -3
- package/dist/security/contentValidator.d.ts +2 -0
- package/dist/security/contentValidator.d.ts.map +1 -1
- package/dist/security/contentValidator.js +115 -4
- package/dist/security/secureYamlParser.d.ts +1 -0
- package/dist/security/secureYamlParser.d.ts.map +1 -1
- package/dist/security/secureYamlParser.js +29 -7
- package/dist/security/securityMonitor.d.ts +1 -1
- package/dist/security/securityMonitor.d.ts.map +1 -1
- package/dist/security/securityMonitor.js +1 -1
- package/dist/security/tokenManager.d.ts +1 -1
- package/dist/security/tokenManager.d.ts.map +1 -1
- package/dist/security/tokenManager.js +30 -10
- package/dist/server/ServerSetup.d.ts +22 -2
- package/dist/server/ServerSetup.d.ts.map +1 -1
- package/dist/server/ServerSetup.js +77 -12
- package/dist/server/tools/AuthTools.d.ts.map +1 -1
- package/dist/server/tools/AuthTools.js +33 -1
- package/dist/server/tools/BuildInfoTools.d.ts +25 -0
- package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
- package/dist/server/tools/BuildInfoTools.js +36 -0
- package/dist/server/tools/CollectionTools.d.ts.map +1 -1
- package/dist/server/tools/CollectionTools.js +55 -46
- package/dist/server/tools/ConfigTools.d.ts.map +1 -1
- package/dist/server/tools/ConfigTools.js +29 -1
- package/dist/server/tools/PersonaTools.d.ts +4 -2
- package/dist/server/tools/PersonaTools.d.ts.map +1 -1
- package/dist/server/tools/PersonaTools.js +5 -152
- package/dist/server/tools/PortfolioTools.d.ts +12 -0
- package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
- package/dist/server/tools/PortfolioTools.js +221 -0
- package/dist/server/tools/index.d.ts +3 -1
- package/dist/server/tools/index.d.ts.map +1 -1
- package/dist/server/tools/index.js +4 -2
- package/dist/server/types.d.ts +40 -5
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/types.js +1 -1
- package/dist/services/BuildInfoService.d.ts +84 -0
- package/dist/services/BuildInfoService.d.ts.map +1 -0
- package/dist/services/BuildInfoService.js +271 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
- package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
- package/dist/tools/portfolio/types.d.ts +41 -0
- package/dist/tools/portfolio/types.d.ts.map +1 -0
- package/dist/tools/portfolio/types.js +15 -0
- package/dist/types/collection.d.ts +51 -0
- package/dist/types/collection.d.ts.map +1 -1
- package/dist/types/collection.js +1 -1
- package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
- package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
- package/dist/utils/EarlyTerminationSearch.js +164 -0
- package/dist/utils/ErrorHandler.d.ts +86 -0
- package/dist/utils/ErrorHandler.d.ts.map +1 -0
- package/dist/utils/ErrorHandler.js +201 -0
- package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
- package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
- package/dist/utils/FileDiscoveryUtil.js +169 -0
- package/dist/utils/GitHubRateLimiter.d.ts +88 -0
- package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
- package/dist/utils/GitHubRateLimiter.js +315 -0
- package/dist/utils/PerformanceMonitor.d.ts +134 -0
- package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
- package/dist/utils/PerformanceMonitor.js +347 -0
- package/dist/utils/RateLimiter.d.ts.map +1 -0
- package/dist/utils/RateLimiter.js +172 -0
- package/dist/utils/SecureDownloader.d.ts +241 -0
- package/dist/utils/SecureDownloader.d.ts.map +1 -0
- package/dist/utils/SecureDownloader.js +759 -0
- package/dist/utils/ToolCache.d.ts +82 -0
- package/dist/utils/ToolCache.d.ts.map +1 -0
- package/dist/utils/ToolCache.js +196 -0
- package/dist/utils/errorCodes.d.ts +136 -0
- package/dist/utils/errorCodes.d.ts.map +1 -0
- package/dist/utils/errorCodes.js +87 -0
- package/dist/utils/index.d.ts +3 -0
- package/dist/utils/index.d.ts.map +1 -1
- package/dist/utils/index.js +4 -1
- package/dist/utils/installation.d.ts +1 -1
- package/dist/utils/installation.d.ts.map +1 -1
- package/dist/utils/installation.js +9 -8
- package/dist/utils/searchUtils.d.ts +31 -0
- package/dist/utils/searchUtils.d.ts.map +1 -1
- package/dist/utils/searchUtils.js +62 -1
- package/package.json +17 -7
- package/dist/config/updateConfig.d.ts +0 -84
- package/dist/config/updateConfig.d.ts.map +0 -1
- package/dist/config/updateConfig.js +0 -148
- package/dist/server/tools/UpdateTools.d.ts +0 -10
- package/dist/server/tools/UpdateTools.d.ts.map +0 -1
- package/dist/server/tools/UpdateTools.js +0 -85
- package/dist/update/BackupManager.d.ts +0 -63
- package/dist/update/BackupManager.d.ts.map +0 -1
- package/dist/update/BackupManager.js +0 -370
- package/dist/update/DependencyChecker.d.ts +0 -41
- package/dist/update/DependencyChecker.d.ts.map +0 -1
- package/dist/update/DependencyChecker.js +0 -132
- package/dist/update/RateLimiter.d.ts.map +0 -1
- package/dist/update/RateLimiter.js +0 -172
- package/dist/update/SignatureVerifier.d.ts +0 -71
- package/dist/update/SignatureVerifier.d.ts.map +0 -1
- package/dist/update/SignatureVerifier.js +0 -214
- package/dist/update/UpdateChecker.d.ts +0 -132
- package/dist/update/UpdateChecker.d.ts.map +0 -1
- package/dist/update/UpdateChecker.js +0 -506
- package/dist/update/UpdateManager.d.ts +0 -60
- package/dist/update/UpdateManager.d.ts.map +0 -1
- package/dist/update/UpdateManager.js +0 -730
- package/dist/update/VersionManager.d.ts +0 -31
- package/dist/update/VersionManager.d.ts.map +0 -1
- package/dist/update/VersionManager.js +0 -181
- package/dist/update/index.d.ts +0 -9
- package/dist/update/index.d.ts.map +0 -1
- package/dist/update/index.js +0 -9
- /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}"]}
|