@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,531 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Comprehensive Performance Benchmarking Suite for DollhouseMCP Indexing System
|
|
3
|
+
*
|
|
4
|
+
* Benchmarks:
|
|
5
|
+
* - Search response times under various loads
|
|
6
|
+
* - Memory usage patterns with large datasets
|
|
7
|
+
* - Cache performance and hit rates
|
|
8
|
+
* - Concurrent operation performance
|
|
9
|
+
* - Index building and rebuilding times
|
|
10
|
+
*/
|
|
11
|
+
import { UnifiedIndexManager } from '../portfolio/UnifiedIndexManager.js';
|
|
12
|
+
import { PortfolioIndexManager } from '../portfolio/PortfolioIndexManager.js';
|
|
13
|
+
import { PerformanceMonitor } from '../utils/PerformanceMonitor.js';
|
|
14
|
+
import { logger } from '../utils/logger.js';
|
|
15
|
+
import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
|
|
16
|
+
export class IndexPerformanceBenchmark {
|
|
17
|
+
unifiedIndexManager;
|
|
18
|
+
performanceMonitor;
|
|
19
|
+
benchmarkResults = [];
|
|
20
|
+
constructor() {
|
|
21
|
+
this.unifiedIndexManager = UnifiedIndexManager.getInstance();
|
|
22
|
+
this.performanceMonitor = PerformanceMonitor.getInstance();
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Run comprehensive performance benchmark suite
|
|
26
|
+
*/
|
|
27
|
+
async runFullBenchmarkSuite() {
|
|
28
|
+
logger.info('Starting comprehensive performance benchmark suite');
|
|
29
|
+
const suiteStartTime = Date.now();
|
|
30
|
+
// Clear caches and reset state
|
|
31
|
+
await this.resetBenchmarkEnvironment();
|
|
32
|
+
const benchmarks = [
|
|
33
|
+
() => this.benchmarkSearchPerformance(),
|
|
34
|
+
() => this.benchmarkMemoryUsage(),
|
|
35
|
+
() => this.benchmarkCachePerformance(),
|
|
36
|
+
() => this.benchmarkConcurrentOperations(),
|
|
37
|
+
() => this.benchmarkLargeDatasetHandling(),
|
|
38
|
+
() => this.benchmarkIndexBuilding(),
|
|
39
|
+
() => this.benchmarkStreamingSearch(),
|
|
40
|
+
() => this.benchmarkLazyLoading()
|
|
41
|
+
];
|
|
42
|
+
// Run each benchmark
|
|
43
|
+
for (const benchmark of benchmarks) {
|
|
44
|
+
try {
|
|
45
|
+
const result = await benchmark();
|
|
46
|
+
this.benchmarkResults.push(result);
|
|
47
|
+
// Allow garbage collection between benchmarks
|
|
48
|
+
await this.waitForGC();
|
|
49
|
+
}
|
|
50
|
+
catch (error) {
|
|
51
|
+
logger.error('Benchmark failed', {
|
|
52
|
+
error: error instanceof Error ? error.message : String(error)
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
const suite = {
|
|
57
|
+
name: 'DollhouseMCP Index Performance Suite',
|
|
58
|
+
results: this.benchmarkResults,
|
|
59
|
+
summary: this.generateSummary(Date.now() - suiteStartTime)
|
|
60
|
+
};
|
|
61
|
+
logger.info('Benchmark suite completed', {
|
|
62
|
+
totalBenchmarks: suite.results.length,
|
|
63
|
+
totalDuration: suite.summary.totalDuration,
|
|
64
|
+
recommendations: suite.summary.recommendations.length
|
|
65
|
+
});
|
|
66
|
+
return suite;
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Benchmark search performance with various query patterns
|
|
70
|
+
*/
|
|
71
|
+
async benchmarkSearchPerformance() {
|
|
72
|
+
const testQueries = [
|
|
73
|
+
'creative',
|
|
74
|
+
'professional assistant',
|
|
75
|
+
'development tools',
|
|
76
|
+
'data analysis',
|
|
77
|
+
'machine learning expert',
|
|
78
|
+
'a very long and complex query that might stress the search system with multiple terms and complex patterns',
|
|
79
|
+
'specific_exact_match',
|
|
80
|
+
'' // Empty query test
|
|
81
|
+
];
|
|
82
|
+
const memoryBefore = process.memoryUsage().heapUsed;
|
|
83
|
+
let peakMemory = memoryBefore;
|
|
84
|
+
const startTime = Date.now();
|
|
85
|
+
let totalOperations = 0;
|
|
86
|
+
let cacheHits = 0;
|
|
87
|
+
for (const query of testQueries) {
|
|
88
|
+
for (let i = 0; i < 10; i++) { // 10 iterations per query
|
|
89
|
+
const options = {
|
|
90
|
+
query,
|
|
91
|
+
includeLocal: true,
|
|
92
|
+
includeGitHub: true,
|
|
93
|
+
includeCollection: false,
|
|
94
|
+
pageSize: 20
|
|
95
|
+
};
|
|
96
|
+
await this.unifiedIndexManager.search(options);
|
|
97
|
+
totalOperations++;
|
|
98
|
+
// Track peak memory
|
|
99
|
+
const currentMemory = process.memoryUsage().heapUsed;
|
|
100
|
+
if (currentMemory > peakMemory) {
|
|
101
|
+
peakMemory = currentMemory;
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
const endTime = Date.now();
|
|
106
|
+
const memoryAfter = process.memoryUsage().heapUsed;
|
|
107
|
+
const duration = endTime - startTime;
|
|
108
|
+
const performanceStats = this.unifiedIndexManager.getPerformanceStats();
|
|
109
|
+
const searchStats = performanceStats.searchStats;
|
|
110
|
+
return {
|
|
111
|
+
name: 'Search Performance',
|
|
112
|
+
duration,
|
|
113
|
+
memoryUsage: {
|
|
114
|
+
before: memoryBefore / (1024 * 1024),
|
|
115
|
+
after: memoryAfter / (1024 * 1024),
|
|
116
|
+
peak: peakMemory / (1024 * 1024)
|
|
117
|
+
},
|
|
118
|
+
throughput: totalOperations / (duration / 1000),
|
|
119
|
+
cacheStats: {
|
|
120
|
+
hitRate: searchStats.cacheHitRate || 0,
|
|
121
|
+
totalOperations
|
|
122
|
+
},
|
|
123
|
+
metadata: {
|
|
124
|
+
avgSearchTime: searchStats.averageTime || 0,
|
|
125
|
+
p95SearchTime: searchStats.p95Time || 0,
|
|
126
|
+
slowQueries: searchStats.slowQueries || 0
|
|
127
|
+
}
|
|
128
|
+
};
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Benchmark memory usage with large result sets
|
|
132
|
+
*/
|
|
133
|
+
async benchmarkMemoryUsage() {
|
|
134
|
+
const memoryBefore = process.memoryUsage().heapUsed;
|
|
135
|
+
let peakMemory = memoryBefore;
|
|
136
|
+
const startTime = Date.now();
|
|
137
|
+
// Create increasingly large search operations
|
|
138
|
+
const largeBatches = [100, 500, 1000, 2000];
|
|
139
|
+
for (const batchSize of largeBatches) {
|
|
140
|
+
const options = {
|
|
141
|
+
query: '', // Empty query to get all results
|
|
142
|
+
includeLocal: true,
|
|
143
|
+
includeGitHub: true,
|
|
144
|
+
includeCollection: true,
|
|
145
|
+
pageSize: batchSize,
|
|
146
|
+
maxResults: batchSize
|
|
147
|
+
};
|
|
148
|
+
await this.unifiedIndexManager.search(options);
|
|
149
|
+
const currentMemory = process.memoryUsage().heapUsed;
|
|
150
|
+
if (currentMemory > peakMemory) {
|
|
151
|
+
peakMemory = currentMemory;
|
|
152
|
+
}
|
|
153
|
+
// Check for memory leaks
|
|
154
|
+
if (currentMemory > memoryBefore * 2) {
|
|
155
|
+
logger.warn('Potential memory leak detected', {
|
|
156
|
+
beforeMB: memoryBefore / (1024 * 1024),
|
|
157
|
+
currentMB: currentMemory / (1024 * 1024),
|
|
158
|
+
batchSize
|
|
159
|
+
});
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
const endTime = Date.now();
|
|
163
|
+
const memoryAfter = process.memoryUsage().heapUsed;
|
|
164
|
+
return {
|
|
165
|
+
name: 'Memory Usage',
|
|
166
|
+
duration: endTime - startTime,
|
|
167
|
+
memoryUsage: {
|
|
168
|
+
before: memoryBefore / (1024 * 1024),
|
|
169
|
+
after: memoryAfter / (1024 * 1024),
|
|
170
|
+
peak: peakMemory / (1024 * 1024)
|
|
171
|
+
},
|
|
172
|
+
metadata: {
|
|
173
|
+
memoryGrowthMB: (memoryAfter - memoryBefore) / (1024 * 1024),
|
|
174
|
+
peakGrowthMB: (peakMemory - memoryBefore) / (1024 * 1024),
|
|
175
|
+
testedBatchSizes: largeBatches
|
|
176
|
+
}
|
|
177
|
+
};
|
|
178
|
+
}
|
|
179
|
+
/**
|
|
180
|
+
* Benchmark cache performance and hit rates
|
|
181
|
+
*/
|
|
182
|
+
async benchmarkCachePerformance() {
|
|
183
|
+
const memoryBefore = process.memoryUsage().heapUsed;
|
|
184
|
+
const startTime = Date.now();
|
|
185
|
+
// Test cache warming and hit rate optimization
|
|
186
|
+
const rawQueries = ['test', 'performance', 'benchmark', 'cache'];
|
|
187
|
+
// DMCP-SEC-004 FIX: Normalize Unicode in all user input
|
|
188
|
+
const testQueries = rawQueries.map(q => {
|
|
189
|
+
const normalized = UnicodeValidator.normalize(q);
|
|
190
|
+
return normalized.isValid ? normalized.normalizedContent : q;
|
|
191
|
+
});
|
|
192
|
+
let totalOperations = 0;
|
|
193
|
+
let cacheHitsBefore = 0;
|
|
194
|
+
// First pass - cache misses expected
|
|
195
|
+
for (const query of testQueries) {
|
|
196
|
+
for (let i = 0; i < 5; i++) {
|
|
197
|
+
await this.unifiedIndexManager.search({ query, pageSize: 10 });
|
|
198
|
+
totalOperations++;
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
const statsAfterFirstPass = this.unifiedIndexManager.getPerformanceStats();
|
|
202
|
+
cacheHitsBefore = statsAfterFirstPass.cacheStats.searchResults.hitCount;
|
|
203
|
+
// Second pass - cache hits expected
|
|
204
|
+
for (const query of testQueries) {
|
|
205
|
+
for (let i = 0; i < 5; i++) {
|
|
206
|
+
await this.unifiedIndexManager.search({ query, pageSize: 10 });
|
|
207
|
+
totalOperations++;
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
const endTime = Date.now();
|
|
211
|
+
const memoryAfter = process.memoryUsage().heapUsed;
|
|
212
|
+
const finalStats = this.unifiedIndexManager.getPerformanceStats();
|
|
213
|
+
const cacheStats = finalStats.cacheStats.searchResults;
|
|
214
|
+
return {
|
|
215
|
+
name: 'Cache Performance',
|
|
216
|
+
duration: endTime - startTime,
|
|
217
|
+
memoryUsage: {
|
|
218
|
+
before: memoryBefore / (1024 * 1024),
|
|
219
|
+
after: memoryAfter / (1024 * 1024),
|
|
220
|
+
peak: memoryAfter / (1024 * 1024)
|
|
221
|
+
},
|
|
222
|
+
cacheStats: {
|
|
223
|
+
hitRate: cacheStats.hitRate,
|
|
224
|
+
totalOperations
|
|
225
|
+
},
|
|
226
|
+
metadata: {
|
|
227
|
+
cacheSize: cacheStats.size,
|
|
228
|
+
evictions: cacheStats.evictionCount,
|
|
229
|
+
hitRateImprovement: cacheStats.hitRate
|
|
230
|
+
}
|
|
231
|
+
};
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Benchmark concurrent search operations
|
|
235
|
+
*/
|
|
236
|
+
async benchmarkConcurrentOperations() {
|
|
237
|
+
const memoryBefore = process.memoryUsage().heapUsed;
|
|
238
|
+
let peakMemory = memoryBefore;
|
|
239
|
+
const startTime = Date.now();
|
|
240
|
+
const concurrencyLevels = [5, 10, 20, 50];
|
|
241
|
+
let totalOperations = 0;
|
|
242
|
+
for (const concurrency of concurrencyLevels) {
|
|
243
|
+
const promises = [];
|
|
244
|
+
for (let i = 0; i < concurrency; i++) {
|
|
245
|
+
const searchPromise = this.unifiedIndexManager.search({
|
|
246
|
+
query: `concurrent_test_${i}`,
|
|
247
|
+
pageSize: 20
|
|
248
|
+
});
|
|
249
|
+
promises.push(searchPromise);
|
|
250
|
+
totalOperations++;
|
|
251
|
+
}
|
|
252
|
+
await Promise.all(promises);
|
|
253
|
+
const currentMemory = process.memoryUsage().heapUsed;
|
|
254
|
+
if (currentMemory > peakMemory) {
|
|
255
|
+
peakMemory = currentMemory;
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
const endTime = Date.now();
|
|
259
|
+
const memoryAfter = process.memoryUsage().heapUsed;
|
|
260
|
+
const duration = endTime - startTime;
|
|
261
|
+
return {
|
|
262
|
+
name: 'Concurrent Operations',
|
|
263
|
+
duration,
|
|
264
|
+
memoryUsage: {
|
|
265
|
+
before: memoryBefore / (1024 * 1024),
|
|
266
|
+
after: memoryAfter / (1024 * 1024),
|
|
267
|
+
peak: peakMemory / (1024 * 1024)
|
|
268
|
+
},
|
|
269
|
+
throughput: totalOperations / (duration / 1000),
|
|
270
|
+
metadata: {
|
|
271
|
+
testedConcurrencyLevels: concurrencyLevels,
|
|
272
|
+
totalConcurrentOps: totalOperations
|
|
273
|
+
}
|
|
274
|
+
};
|
|
275
|
+
}
|
|
276
|
+
/**
|
|
277
|
+
* Benchmark large dataset handling (simulated)
|
|
278
|
+
*/
|
|
279
|
+
async benchmarkLargeDatasetHandling() {
|
|
280
|
+
const memoryBefore = process.memoryUsage().heapUsed;
|
|
281
|
+
let peakMemory = memoryBefore;
|
|
282
|
+
const startTime = Date.now();
|
|
283
|
+
// Simulate searches that would stress a large dataset
|
|
284
|
+
const stressTestQueries = [
|
|
285
|
+
{ query: '', pageSize: 1000 }, // Get all results
|
|
286
|
+
{ query: 'common_term', pageSize: 500 }, // High result count
|
|
287
|
+
{ query: 'very_specific_unique_term', pageSize: 100 }, // Low result count
|
|
288
|
+
{ query: 'partial_match_test', pageSize: 200 } // Medium result count
|
|
289
|
+
];
|
|
290
|
+
let totalResults = 0;
|
|
291
|
+
for (const testCase of stressTestQueries) {
|
|
292
|
+
const results = await this.unifiedIndexManager.search({
|
|
293
|
+
query: testCase.query,
|
|
294
|
+
pageSize: testCase.pageSize,
|
|
295
|
+
includeLocal: true,
|
|
296
|
+
includeGitHub: true,
|
|
297
|
+
includeCollection: true
|
|
298
|
+
});
|
|
299
|
+
totalResults += results.length;
|
|
300
|
+
const currentMemory = process.memoryUsage().heapUsed;
|
|
301
|
+
if (currentMemory > peakMemory) {
|
|
302
|
+
peakMemory = currentMemory;
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
const endTime = Date.now();
|
|
306
|
+
const memoryAfter = process.memoryUsage().heapUsed;
|
|
307
|
+
return {
|
|
308
|
+
name: 'Large Dataset Handling',
|
|
309
|
+
duration: endTime - startTime,
|
|
310
|
+
memoryUsage: {
|
|
311
|
+
before: memoryBefore / (1024 * 1024),
|
|
312
|
+
after: memoryAfter / (1024 * 1024),
|
|
313
|
+
peak: peakMemory / (1024 * 1024)
|
|
314
|
+
},
|
|
315
|
+
metadata: {
|
|
316
|
+
totalResultsProcessed: totalResults,
|
|
317
|
+
averageResultsPerQuery: totalResults / stressTestQueries.length,
|
|
318
|
+
memoryPerResult: (peakMemory - memoryBefore) / totalResults
|
|
319
|
+
}
|
|
320
|
+
};
|
|
321
|
+
}
|
|
322
|
+
/**
|
|
323
|
+
* Benchmark index building performance
|
|
324
|
+
*/
|
|
325
|
+
async benchmarkIndexBuilding() {
|
|
326
|
+
const memoryBefore = process.memoryUsage().heapUsed;
|
|
327
|
+
const startTime = Date.now();
|
|
328
|
+
// Force rebuild of all indexes
|
|
329
|
+
await this.unifiedIndexManager.rebuildAll();
|
|
330
|
+
const endTime = Date.now();
|
|
331
|
+
const memoryAfter = process.memoryUsage().heapUsed;
|
|
332
|
+
const localStats = await PortfolioIndexManager.getInstance().getStats();
|
|
333
|
+
return {
|
|
334
|
+
name: 'Index Building',
|
|
335
|
+
duration: endTime - startTime,
|
|
336
|
+
memoryUsage: {
|
|
337
|
+
before: memoryBefore / (1024 * 1024),
|
|
338
|
+
after: memoryAfter / (1024 * 1024),
|
|
339
|
+
peak: memoryAfter / (1024 * 1024)
|
|
340
|
+
},
|
|
341
|
+
metadata: {
|
|
342
|
+
totalElementsIndexed: localStats.totalElements,
|
|
343
|
+
indexingRate: localStats.totalElements / ((endTime - startTime) / 1000),
|
|
344
|
+
isStale: localStats.isStale
|
|
345
|
+
}
|
|
346
|
+
};
|
|
347
|
+
}
|
|
348
|
+
/**
|
|
349
|
+
* Benchmark streaming search performance
|
|
350
|
+
*/
|
|
351
|
+
async benchmarkStreamingSearch() {
|
|
352
|
+
const memoryBefore = process.memoryUsage().heapUsed;
|
|
353
|
+
let peakMemory = memoryBefore;
|
|
354
|
+
const startTime = Date.now();
|
|
355
|
+
const streamingQueries = [
|
|
356
|
+
{ query: 'test', maxResults: 100 },
|
|
357
|
+
{ query: 'assistant', maxResults: 200 },
|
|
358
|
+
{ query: 'creative', maxResults: 150 }
|
|
359
|
+
];
|
|
360
|
+
let totalResults = 0;
|
|
361
|
+
for (const testCase of streamingQueries) {
|
|
362
|
+
const results = await this.unifiedIndexManager.search({
|
|
363
|
+
query: testCase.query,
|
|
364
|
+
streamResults: true,
|
|
365
|
+
maxResults: testCase.maxResults,
|
|
366
|
+
includeLocal: true,
|
|
367
|
+
includeGitHub: true
|
|
368
|
+
});
|
|
369
|
+
totalResults += results.length;
|
|
370
|
+
const currentMemory = process.memoryUsage().heapUsed;
|
|
371
|
+
if (currentMemory > peakMemory) {
|
|
372
|
+
peakMemory = currentMemory;
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
const endTime = Date.now();
|
|
376
|
+
const memoryAfter = process.memoryUsage().heapUsed;
|
|
377
|
+
return {
|
|
378
|
+
name: 'Streaming Search',
|
|
379
|
+
duration: endTime - startTime,
|
|
380
|
+
memoryUsage: {
|
|
381
|
+
before: memoryBefore / (1024 * 1024),
|
|
382
|
+
after: memoryAfter / (1024 * 1024),
|
|
383
|
+
peak: peakMemory / (1024 * 1024)
|
|
384
|
+
},
|
|
385
|
+
throughput: totalResults / ((endTime - startTime) / 1000),
|
|
386
|
+
metadata: {
|
|
387
|
+
totalStreamedResults: totalResults,
|
|
388
|
+
averageMemoryPerResult: (peakMemory - memoryBefore) / totalResults
|
|
389
|
+
}
|
|
390
|
+
};
|
|
391
|
+
}
|
|
392
|
+
/**
|
|
393
|
+
* Benchmark lazy loading performance
|
|
394
|
+
*/
|
|
395
|
+
async benchmarkLazyLoading() {
|
|
396
|
+
const memoryBefore = process.memoryUsage().heapUsed;
|
|
397
|
+
const startTime = Date.now();
|
|
398
|
+
// Test lazy loading with different configurations
|
|
399
|
+
const lazyLoadTests = [
|
|
400
|
+
{ query: 'lazy_test_1', lazyLoad: true, includeLocal: true, includeGitHub: false, includeCollection: false },
|
|
401
|
+
{ query: 'lazy_test_2', lazyLoad: true, includeLocal: false, includeGitHub: true, includeCollection: false },
|
|
402
|
+
{ query: 'lazy_test_3', lazyLoad: true, includeLocal: true, includeGitHub: true, includeCollection: true },
|
|
403
|
+
{ query: 'lazy_test_4', lazyLoad: false, includeLocal: true, includeGitHub: true, includeCollection: true }
|
|
404
|
+
];
|
|
405
|
+
let totalOperations = 0;
|
|
406
|
+
for (const testCase of lazyLoadTests) {
|
|
407
|
+
await this.unifiedIndexManager.search(testCase);
|
|
408
|
+
totalOperations++;
|
|
409
|
+
}
|
|
410
|
+
const endTime = Date.now();
|
|
411
|
+
const memoryAfter = process.memoryUsage().heapUsed;
|
|
412
|
+
return {
|
|
413
|
+
name: 'Lazy Loading',
|
|
414
|
+
duration: endTime - startTime,
|
|
415
|
+
memoryUsage: {
|
|
416
|
+
before: memoryBefore / (1024 * 1024),
|
|
417
|
+
after: memoryAfter / (1024 * 1024),
|
|
418
|
+
peak: memoryAfter / (1024 * 1024)
|
|
419
|
+
},
|
|
420
|
+
throughput: totalOperations / ((endTime - startTime) / 1000),
|
|
421
|
+
metadata: {
|
|
422
|
+
totalLazyOperations: totalOperations,
|
|
423
|
+
averageOperationTime: (endTime - startTime) / totalOperations
|
|
424
|
+
}
|
|
425
|
+
};
|
|
426
|
+
}
|
|
427
|
+
/**
|
|
428
|
+
* Reset benchmark environment
|
|
429
|
+
*/
|
|
430
|
+
async resetBenchmarkEnvironment() {
|
|
431
|
+
await this.unifiedIndexManager.rebuildAll();
|
|
432
|
+
this.performanceMonitor.reset();
|
|
433
|
+
this.benchmarkResults = [];
|
|
434
|
+
// Force garbage collection if available
|
|
435
|
+
if (global.gc) {
|
|
436
|
+
global.gc();
|
|
437
|
+
}
|
|
438
|
+
await this.waitForGC();
|
|
439
|
+
}
|
|
440
|
+
/**
|
|
441
|
+
* Wait for garbage collection to complete
|
|
442
|
+
*/
|
|
443
|
+
async waitForGC() {
|
|
444
|
+
return new Promise(resolve => {
|
|
445
|
+
setImmediate(() => {
|
|
446
|
+
if (global.gc) {
|
|
447
|
+
global.gc();
|
|
448
|
+
}
|
|
449
|
+
setTimeout(resolve, 100); // Wait 100ms for GC to complete
|
|
450
|
+
});
|
|
451
|
+
});
|
|
452
|
+
}
|
|
453
|
+
/**
|
|
454
|
+
* Generate benchmark summary and recommendations
|
|
455
|
+
*/
|
|
456
|
+
generateSummary(totalDuration) {
|
|
457
|
+
const recommendations = [];
|
|
458
|
+
let totalMemoryUsage = 0;
|
|
459
|
+
let peakMemoryUsage = 0;
|
|
460
|
+
// Analyze results and generate recommendations
|
|
461
|
+
for (const result of this.benchmarkResults) {
|
|
462
|
+
totalMemoryUsage += result.memoryUsage.after - result.memoryUsage.before;
|
|
463
|
+
peakMemoryUsage = Math.max(peakMemoryUsage, result.memoryUsage.peak);
|
|
464
|
+
// Check for performance issues
|
|
465
|
+
if (result.name === 'Search Performance' && result.metadata?.avgSearchTime > 100) {
|
|
466
|
+
recommendations.push('Average search time exceeds 100ms. Consider query optimization or increased caching.');
|
|
467
|
+
}
|
|
468
|
+
if (result.memoryUsage.peak > 200) {
|
|
469
|
+
recommendations.push(`High memory usage detected in ${result.name} (${result.memoryUsage.peak.toFixed(1)}MB). Consider memory optimization.`);
|
|
470
|
+
}
|
|
471
|
+
if (result.cacheStats?.hitRate && result.cacheStats.hitRate < 0.5) {
|
|
472
|
+
recommendations.push(`Low cache hit rate in ${result.name} (${result.cacheStats.hitRate.toFixed(2)}). Consider cache size increase or TTL adjustment.`);
|
|
473
|
+
}
|
|
474
|
+
if (result.throughput && result.throughput < 10) {
|
|
475
|
+
recommendations.push(`Low throughput in ${result.name} (${result.throughput.toFixed(1)} ops/sec). Consider performance optimization.`);
|
|
476
|
+
}
|
|
477
|
+
}
|
|
478
|
+
// General recommendations
|
|
479
|
+
if (peakMemoryUsage > 50) {
|
|
480
|
+
recommendations.push('Peak memory usage exceeds 50MB. Consider implementing more aggressive memory cleanup.');
|
|
481
|
+
}
|
|
482
|
+
if (totalDuration > 30000) {
|
|
483
|
+
recommendations.push('Total benchmark duration exceeds 30 seconds. Consider performance optimizations.');
|
|
484
|
+
}
|
|
485
|
+
return {
|
|
486
|
+
totalDuration,
|
|
487
|
+
averageMemoryUsage: totalMemoryUsage / this.benchmarkResults.length,
|
|
488
|
+
peakMemoryUsage,
|
|
489
|
+
recommendations
|
|
490
|
+
};
|
|
491
|
+
}
|
|
492
|
+
/**
|
|
493
|
+
* Export benchmark results to JSON
|
|
494
|
+
*/
|
|
495
|
+
exportResults(suite) {
|
|
496
|
+
return JSON.stringify(suite, null, 2);
|
|
497
|
+
}
|
|
498
|
+
/**
|
|
499
|
+
* Generate benchmark report
|
|
500
|
+
*/
|
|
501
|
+
generateReport(suite) {
|
|
502
|
+
let report = `# DollhouseMCP Index Performance Benchmark Report\n\n`;
|
|
503
|
+
report += `**Generated:** ${new Date().toISOString()}\n`;
|
|
504
|
+
report += `**Total Duration:** ${suite.summary.totalDuration}ms\n`;
|
|
505
|
+
report += `**Peak Memory Usage:** ${suite.summary.peakMemoryUsage.toFixed(1)}MB\n\n`;
|
|
506
|
+
report += `## Benchmark Results\n\n`;
|
|
507
|
+
for (const result of suite.results) {
|
|
508
|
+
report += `### ${result.name}\n`;
|
|
509
|
+
report += `- **Duration:** ${result.duration}ms\n`;
|
|
510
|
+
report += `- **Memory Usage:** ${result.memoryUsage.before.toFixed(1)}MB → ${result.memoryUsage.after.toFixed(1)}MB (Peak: ${result.memoryUsage.peak.toFixed(1)}MB)\n`;
|
|
511
|
+
if (result.throughput) {
|
|
512
|
+
report += `- **Throughput:** ${result.throughput.toFixed(1)} ops/sec\n`;
|
|
513
|
+
}
|
|
514
|
+
if (result.cacheStats) {
|
|
515
|
+
report += `- **Cache Hit Rate:** ${result.cacheStats.hitRate.toFixed(2)}\n`;
|
|
516
|
+
}
|
|
517
|
+
report += `\n`;
|
|
518
|
+
}
|
|
519
|
+
report += `## Recommendations\n\n`;
|
|
520
|
+
if (suite.summary.recommendations.length === 0) {
|
|
521
|
+
report += `No performance issues detected. System is performing within acceptable parameters.\n`;
|
|
522
|
+
}
|
|
523
|
+
else {
|
|
524
|
+
for (const recommendation of suite.summary.recommendations) {
|
|
525
|
+
report += `- ${recommendation}\n`;
|
|
526
|
+
}
|
|
527
|
+
}
|
|
528
|
+
return report;
|
|
529
|
+
}
|
|
530
|
+
}
|
|
531
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"IndexPerformanceBenchmark.js","sourceRoot":"","sources":["../../src/benchmarks/IndexPerformanceBenchmark.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Comprehensive Performance Benchmarking Suite for DollhouseMCP Indexing System\n * \n * Benchmarks:\n * - Search response times under various loads\n * - Memory usage patterns with large datasets\n * - Cache performance and hit rates\n * - Concurrent operation performance\n * - Index building and rebuilding times\n */\n\nimport { UnifiedIndexManager, UnifiedSearchOptions } from '../portfolio/UnifiedIndexManager.js';\nimport { PortfolioIndexManager } from '../portfolio/PortfolioIndexManager.js';\nimport { CollectionIndexCache } from '../cache/CollectionIndexCache.js';\nimport { PerformanceMonitor } from '../utils/PerformanceMonitor.js';\nimport { GitHubClient } from '../collection/GitHubClient.js';\nimport { APICache } from '../cache/APICache.js';\nimport { logger } from '../utils/logger.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\n\nexport interface BenchmarkResult {\n  name: string;\n  duration: number;\n  memoryUsage: {\n    before: number;\n    after: number;\n    peak: number;\n  };\n  throughput?: number; // operations per second\n  cacheStats?: {\n    hitRate: number;\n    totalOperations: number;\n  };\n  metadata?: any;\n}\n\nexport interface BenchmarkSuite {\n  name: string;\n  results: BenchmarkResult[];\n  summary: {\n    totalDuration: number;\n    averageMemoryUsage: number;\n    peakMemoryUsage: number;\n    recommendations: string[];\n  };\n}\n\nexport class IndexPerformanceBenchmark {\n  private unifiedIndexManager: UnifiedIndexManager;\n  private performanceMonitor: PerformanceMonitor;\n  private benchmarkResults: BenchmarkResult[] = [];\n\n  constructor() {\n    this.unifiedIndexManager = UnifiedIndexManager.getInstance();\n    this.performanceMonitor = PerformanceMonitor.getInstance();\n  }\n\n  /**\n   * Run comprehensive performance benchmark suite\n   */\n  async runFullBenchmarkSuite(): Promise<BenchmarkSuite> {\n    logger.info('Starting comprehensive performance benchmark suite');\n    const suiteStartTime = Date.now();\n\n    // Clear caches and reset state\n    await this.resetBenchmarkEnvironment();\n\n    const benchmarks = [\n      () => this.benchmarkSearchPerformance(),\n      () => this.benchmarkMemoryUsage(),\n      () => this.benchmarkCachePerformance(),\n      () => this.benchmarkConcurrentOperations(),\n      () => this.benchmarkLargeDatasetHandling(),\n      () => this.benchmarkIndexBuilding(),\n      () => this.benchmarkStreamingSearch(),\n      () => this.benchmarkLazyLoading()\n    ];\n\n    // Run each benchmark\n    for (const benchmark of benchmarks) {\n      try {\n        const result = await benchmark();\n        this.benchmarkResults.push(result);\n        \n        // Allow garbage collection between benchmarks\n        await this.waitForGC();\n      } catch (error) {\n        logger.error('Benchmark failed', {\n          error: error instanceof Error ? error.message : String(error)\n        });\n      }\n    }\n\n    const suite: BenchmarkSuite = {\n      name: 'DollhouseMCP Index Performance Suite',\n      results: this.benchmarkResults,\n      summary: this.generateSummary(Date.now() - suiteStartTime)\n    };\n\n    logger.info('Benchmark suite completed', {\n      totalBenchmarks: suite.results.length,\n      totalDuration: suite.summary.totalDuration,\n      recommendations: suite.summary.recommendations.length\n    });\n\n    return suite;\n  }\n\n  /**\n   * Benchmark search performance with various query patterns\n   */\n  private async benchmarkSearchPerformance(): Promise<BenchmarkResult> {\n    const testQueries = [\n      'creative',\n      'professional assistant',\n      'development tools',\n      'data analysis',\n      'machine learning expert',\n      'a very long and complex query that might stress the search system with multiple terms and complex patterns',\n      'specific_exact_match',\n      ''  // Empty query test\n    ];\n\n    const memoryBefore = process.memoryUsage().heapUsed;\n    let peakMemory = memoryBefore;\n    const startTime = Date.now();\n\n    let totalOperations = 0;\n    let cacheHits = 0;\n\n    for (const query of testQueries) {\n      for (let i = 0; i < 10; i++) { // 10 iterations per query\n        const options: UnifiedSearchOptions = {\n          query,\n          includeLocal: true,\n          includeGitHub: true,\n          includeCollection: false,\n          pageSize: 20\n        };\n\n        await this.unifiedIndexManager.search(options);\n        totalOperations++;\n\n        // Track peak memory\n        const currentMemory = process.memoryUsage().heapUsed;\n        if (currentMemory > peakMemory) {\n          peakMemory = currentMemory;\n        }\n      }\n    }\n\n    const endTime = Date.now();\n    const memoryAfter = process.memoryUsage().heapUsed;\n    const duration = endTime - startTime;\n\n    const performanceStats = this.unifiedIndexManager.getPerformanceStats();\n    const searchStats = performanceStats.searchStats;\n\n    return {\n      name: 'Search Performance',\n      duration,\n      memoryUsage: {\n        before: memoryBefore / (1024 * 1024),\n        after: memoryAfter / (1024 * 1024),\n        peak: peakMemory / (1024 * 1024)\n      },\n      throughput: totalOperations / (duration / 1000),\n      cacheStats: {\n        hitRate: searchStats.cacheHitRate || 0,\n        totalOperations\n      },\n      metadata: {\n        avgSearchTime: searchStats.averageTime || 0,\n        p95SearchTime: searchStats.p95Time || 0,\n        slowQueries: searchStats.slowQueries || 0\n      }\n    };\n  }\n\n  /**\n   * Benchmark memory usage with large result sets\n   */\n  private async benchmarkMemoryUsage(): Promise<BenchmarkResult> {\n    const memoryBefore = process.memoryUsage().heapUsed;\n    let peakMemory = memoryBefore;\n    const startTime = Date.now();\n\n    // Create increasingly large search operations\n    const largeBatches = [100, 500, 1000, 2000];\n    \n    for (const batchSize of largeBatches) {\n      const options: UnifiedSearchOptions = {\n        query: '',  // Empty query to get all results\n        includeLocal: true,\n        includeGitHub: true,\n        includeCollection: true,\n        pageSize: batchSize,\n        maxResults: batchSize\n      };\n\n      await this.unifiedIndexManager.search(options);\n\n      const currentMemory = process.memoryUsage().heapUsed;\n      if (currentMemory > peakMemory) {\n        peakMemory = currentMemory;\n      }\n\n      // Check for memory leaks\n      if (currentMemory > memoryBefore * 2) {\n        logger.warn('Potential memory leak detected', {\n          beforeMB: memoryBefore / (1024 * 1024),\n          currentMB: currentMemory / (1024 * 1024),\n          batchSize\n        });\n      }\n    }\n\n    const endTime = Date.now();\n    const memoryAfter = process.memoryUsage().heapUsed;\n\n    return {\n      name: 'Memory Usage',\n      duration: endTime - startTime,\n      memoryUsage: {\n        before: memoryBefore / (1024 * 1024),\n        after: memoryAfter / (1024 * 1024),\n        peak: peakMemory / (1024 * 1024)\n      },\n      metadata: {\n        memoryGrowthMB: (memoryAfter - memoryBefore) / (1024 * 1024),\n        peakGrowthMB: (peakMemory - memoryBefore) / (1024 * 1024),\n        testedBatchSizes: largeBatches\n      }\n    };\n  }\n\n  /**\n   * Benchmark cache performance and hit rates\n   */\n  private async benchmarkCachePerformance(): Promise<BenchmarkResult> {\n    const memoryBefore = process.memoryUsage().heapUsed;\n    const startTime = Date.now();\n\n    // Test cache warming and hit rate optimization\n    const rawQueries = ['test', 'performance', 'benchmark', 'cache'];\n    // DMCP-SEC-004 FIX: Normalize Unicode in all user input\n    const testQueries = rawQueries.map(q => {\n      const normalized = UnicodeValidator.normalize(q);\n      return normalized.isValid ? normalized.normalizedContent : q;\n    });\n    let totalOperations = 0;\n    let cacheHitsBefore = 0;\n\n    // First pass - cache misses expected\n    for (const query of testQueries) {\n      for (let i = 0; i < 5; i++) {\n        await this.unifiedIndexManager.search({ query, pageSize: 10 });\n        totalOperations++;\n      }\n    }\n\n    const statsAfterFirstPass = this.unifiedIndexManager.getPerformanceStats();\n    cacheHitsBefore = statsAfterFirstPass.cacheStats.searchResults.hitCount;\n\n    // Second pass - cache hits expected\n    for (const query of testQueries) {\n      for (let i = 0; i < 5; i++) {\n        await this.unifiedIndexManager.search({ query, pageSize: 10 });\n        totalOperations++;\n      }\n    }\n\n    const endTime = Date.now();\n    const memoryAfter = process.memoryUsage().heapUsed;\n\n    const finalStats = this.unifiedIndexManager.getPerformanceStats();\n    const cacheStats = finalStats.cacheStats.searchResults;\n\n    return {\n      name: 'Cache Performance',\n      duration: endTime - startTime,\n      memoryUsage: {\n        before: memoryBefore / (1024 * 1024),\n        after: memoryAfter / (1024 * 1024),\n        peak: memoryAfter / (1024 * 1024)\n      },\n      cacheStats: {\n        hitRate: cacheStats.hitRate,\n        totalOperations\n      },\n      metadata: {\n        cacheSize: cacheStats.size,\n        evictions: cacheStats.evictionCount,\n        hitRateImprovement: cacheStats.hitRate\n      }\n    };\n  }\n\n  /**\n   * Benchmark concurrent search operations\n   */\n  private async benchmarkConcurrentOperations(): Promise<BenchmarkResult> {\n    const memoryBefore = process.memoryUsage().heapUsed;\n    let peakMemory = memoryBefore;\n    const startTime = Date.now();\n\n    const concurrencyLevels = [5, 10, 20, 50];\n    let totalOperations = 0;\n\n    for (const concurrency of concurrencyLevels) {\n      const promises = [];\n\n      for (let i = 0; i < concurrency; i++) {\n        const searchPromise = this.unifiedIndexManager.search({\n          query: `concurrent_test_${i}`,\n          pageSize: 20\n        });\n        promises.push(searchPromise);\n        totalOperations++;\n      }\n\n      await Promise.all(promises);\n\n      const currentMemory = process.memoryUsage().heapUsed;\n      if (currentMemory > peakMemory) {\n        peakMemory = currentMemory;\n      }\n    }\n\n    const endTime = Date.now();\n    const memoryAfter = process.memoryUsage().heapUsed;\n    const duration = endTime - startTime;\n\n    return {\n      name: 'Concurrent Operations',\n      duration,\n      memoryUsage: {\n        before: memoryBefore / (1024 * 1024),\n        after: memoryAfter / (1024 * 1024),\n        peak: peakMemory / (1024 * 1024)\n      },\n      throughput: totalOperations / (duration / 1000),\n      metadata: {\n        testedConcurrencyLevels: concurrencyLevels,\n        totalConcurrentOps: totalOperations\n      }\n    };\n  }\n\n  /**\n   * Benchmark large dataset handling (simulated)\n   */\n  private async benchmarkLargeDatasetHandling(): Promise<BenchmarkResult> {\n    const memoryBefore = process.memoryUsage().heapUsed;\n    let peakMemory = memoryBefore;\n    const startTime = Date.now();\n\n    // Simulate searches that would stress a large dataset\n    const stressTestQueries = [\n      { query: '', pageSize: 1000 }, // Get all results\n      { query: 'common_term', pageSize: 500 }, // High result count\n      { query: 'very_specific_unique_term', pageSize: 100 }, // Low result count\n      { query: 'partial_match_test', pageSize: 200 } // Medium result count\n    ];\n\n    let totalResults = 0;\n\n    for (const testCase of stressTestQueries) {\n      const results = await this.unifiedIndexManager.search({\n        query: testCase.query,\n        pageSize: testCase.pageSize,\n        includeLocal: true,\n        includeGitHub: true,\n        includeCollection: true\n      });\n\n      totalResults += results.length;\n\n      const currentMemory = process.memoryUsage().heapUsed;\n      if (currentMemory > peakMemory) {\n        peakMemory = currentMemory;\n      }\n    }\n\n    const endTime = Date.now();\n    const memoryAfter = process.memoryUsage().heapUsed;\n\n    return {\n      name: 'Large Dataset Handling',\n      duration: endTime - startTime,\n      memoryUsage: {\n        before: memoryBefore / (1024 * 1024),\n        after: memoryAfter / (1024 * 1024),\n        peak: peakMemory / (1024 * 1024)\n      },\n      metadata: {\n        totalResultsProcessed: totalResults,\n        averageResultsPerQuery: totalResults / stressTestQueries.length,\n        memoryPerResult: (peakMemory - memoryBefore) / totalResults\n      }\n    };\n  }\n\n  /**\n   * Benchmark index building performance\n   */\n  private async benchmarkIndexBuilding(): Promise<BenchmarkResult> {\n    const memoryBefore = process.memoryUsage().heapUsed;\n    const startTime = Date.now();\n\n    // Force rebuild of all indexes\n    await this.unifiedIndexManager.rebuildAll();\n\n    const endTime = Date.now();\n    const memoryAfter = process.memoryUsage().heapUsed;\n\n    const localStats = await PortfolioIndexManager.getInstance().getStats();\n\n    return {\n      name: 'Index Building',\n      duration: endTime - startTime,\n      memoryUsage: {\n        before: memoryBefore / (1024 * 1024),\n        after: memoryAfter / (1024 * 1024),\n        peak: memoryAfter / (1024 * 1024)\n      },\n      metadata: {\n        totalElementsIndexed: localStats.totalElements,\n        indexingRate: localStats.totalElements / ((endTime - startTime) / 1000),\n        isStale: localStats.isStale\n      }\n    };\n  }\n\n  /**\n   * Benchmark streaming search performance\n   */\n  private async benchmarkStreamingSearch(): Promise<BenchmarkResult> {\n    const memoryBefore = process.memoryUsage().heapUsed;\n    let peakMemory = memoryBefore;\n    const startTime = Date.now();\n\n    const streamingQueries = [\n      { query: 'test', maxResults: 100 },\n      { query: 'assistant', maxResults: 200 },\n      { query: 'creative', maxResults: 150 }\n    ];\n\n    let totalResults = 0;\n\n    for (const testCase of streamingQueries) {\n      const results = await this.unifiedIndexManager.search({\n        query: testCase.query,\n        streamResults: true,\n        maxResults: testCase.maxResults,\n        includeLocal: true,\n        includeGitHub: true\n      });\n\n      totalResults += results.length;\n\n      const currentMemory = process.memoryUsage().heapUsed;\n      if (currentMemory > peakMemory) {\n        peakMemory = currentMemory;\n      }\n    }\n\n    const endTime = Date.now();\n    const memoryAfter = process.memoryUsage().heapUsed;\n\n    return {\n      name: 'Streaming Search',\n      duration: endTime - startTime,\n      memoryUsage: {\n        before: memoryBefore / (1024 * 1024),\n        after: memoryAfter / (1024 * 1024),\n        peak: peakMemory / (1024 * 1024)\n      },\n      throughput: totalResults / ((endTime - startTime) / 1000),\n      metadata: {\n        totalStreamedResults: totalResults,\n        averageMemoryPerResult: (peakMemory - memoryBefore) / totalResults\n      }\n    };\n  }\n\n  /**\n   * Benchmark lazy loading performance\n   */\n  private async benchmarkLazyLoading(): Promise<BenchmarkResult> {\n    const memoryBefore = process.memoryUsage().heapUsed;\n    const startTime = Date.now();\n\n    // Test lazy loading with different configurations\n    const lazyLoadTests = [\n      { query: 'lazy_test_1', lazyLoad: true, includeLocal: true, includeGitHub: false, includeCollection: false },\n      { query: 'lazy_test_2', lazyLoad: true, includeLocal: false, includeGitHub: true, includeCollection: false },\n      { query: 'lazy_test_3', lazyLoad: true, includeLocal: true, includeGitHub: true, includeCollection: true },\n      { query: 'lazy_test_4', lazyLoad: false, includeLocal: true, includeGitHub: true, includeCollection: true }\n    ];\n\n    let totalOperations = 0;\n\n    for (const testCase of lazyLoadTests) {\n      await this.unifiedIndexManager.search(testCase);\n      totalOperations++;\n    }\n\n    const endTime = Date.now();\n    const memoryAfter = process.memoryUsage().heapUsed;\n\n    return {\n      name: 'Lazy Loading',\n      duration: endTime - startTime,\n      memoryUsage: {\n        before: memoryBefore / (1024 * 1024),\n        after: memoryAfter / (1024 * 1024),\n        peak: memoryAfter / (1024 * 1024)\n      },\n      throughput: totalOperations / ((endTime - startTime) / 1000),\n      metadata: {\n        totalLazyOperations: totalOperations,\n        averageOperationTime: (endTime - startTime) / totalOperations\n      }\n    };\n  }\n\n  /**\n   * Reset benchmark environment\n   */\n  private async resetBenchmarkEnvironment(): Promise<void> {\n    await this.unifiedIndexManager.rebuildAll();\n    this.performanceMonitor.reset();\n    this.benchmarkResults = [];\n\n    // Force garbage collection if available\n    if (global.gc) {\n      global.gc();\n    }\n\n    await this.waitForGC();\n  }\n\n  /**\n   * Wait for garbage collection to complete\n   */\n  private async waitForGC(): Promise<void> {\n    return new Promise(resolve => {\n      setImmediate(() => {\n        if (global.gc) {\n          global.gc();\n        }\n        setTimeout(resolve, 100); // Wait 100ms for GC to complete\n      });\n    });\n  }\n\n  /**\n   * Generate benchmark summary and recommendations\n   */\n  private generateSummary(totalDuration: number): BenchmarkSuite['summary'] {\n    const recommendations: string[] = [];\n    let totalMemoryUsage = 0;\n    let peakMemoryUsage = 0;\n\n    // Analyze results and generate recommendations\n    for (const result of this.benchmarkResults) {\n      totalMemoryUsage += result.memoryUsage.after - result.memoryUsage.before;\n      peakMemoryUsage = Math.max(peakMemoryUsage, result.memoryUsage.peak);\n\n      // Check for performance issues\n      if (result.name === 'Search Performance' && result.metadata?.avgSearchTime > 100) {\n        recommendations.push('Average search time exceeds 100ms. Consider query optimization or increased caching.');\n      }\n\n      if (result.memoryUsage.peak > 200) {\n        recommendations.push(`High memory usage detected in ${result.name} (${result.memoryUsage.peak.toFixed(1)}MB). Consider memory optimization.`);\n      }\n\n      if (result.cacheStats?.hitRate && result.cacheStats.hitRate < 0.5) {\n        recommendations.push(`Low cache hit rate in ${result.name} (${result.cacheStats.hitRate.toFixed(2)}). Consider cache size increase or TTL adjustment.`);\n      }\n\n      if (result.throughput && result.throughput < 10) {\n        recommendations.push(`Low throughput in ${result.name} (${result.throughput.toFixed(1)} ops/sec). Consider performance optimization.`);\n      }\n    }\n\n    // General recommendations\n    if (peakMemoryUsage > 50) {\n      recommendations.push('Peak memory usage exceeds 50MB. Consider implementing more aggressive memory cleanup.');\n    }\n\n    if (totalDuration > 30000) {\n      recommendations.push('Total benchmark duration exceeds 30 seconds. Consider performance optimizations.');\n    }\n\n    return {\n      totalDuration,\n      averageMemoryUsage: totalMemoryUsage / this.benchmarkResults.length,\n      peakMemoryUsage,\n      recommendations\n    };\n  }\n\n  /**\n   * Export benchmark results to JSON\n   */\n  exportResults(suite: BenchmarkSuite): string {\n    return JSON.stringify(suite, null, 2);\n  }\n\n  /**\n   * Generate benchmark report\n   */\n  generateReport(suite: BenchmarkSuite): string {\n    let report = `# DollhouseMCP Index Performance Benchmark Report\\n\\n`;\n    report += `**Generated:** ${new Date().toISOString()}\\n`;\n    report += `**Total Duration:** ${suite.summary.totalDuration}ms\\n`;\n    report += `**Peak Memory Usage:** ${suite.summary.peakMemoryUsage.toFixed(1)}MB\\n\\n`;\n\n    report += `## Benchmark Results\\n\\n`;\n\n    for (const result of suite.results) {\n      report += `### ${result.name}\\n`;\n      report += `- **Duration:** ${result.duration}ms\\n`;\n      report += `- **Memory Usage:** ${result.memoryUsage.before.toFixed(1)}MB → ${result.memoryUsage.after.toFixed(1)}MB (Peak: ${result.memoryUsage.peak.toFixed(1)}MB)\\n`;\n      \n      if (result.throughput) {\n        report += `- **Throughput:** ${result.throughput.toFixed(1)} ops/sec\\n`;\n      }\n      \n      if (result.cacheStats) {\n        report += `- **Cache Hit Rate:** ${result.cacheStats.hitRate.toFixed(2)}\\n`;\n      }\n      \n      report += `\\n`;\n    }\n\n    report += `## Recommendations\\n\\n`;\n    \n    if (suite.summary.recommendations.length === 0) {\n      report += `No performance issues detected. System is performing within acceptable parameters.\\n`;\n    } else {\n      for (const recommendation of suite.summary.recommendations) {\n        report += `- ${recommendation}\\n`;\n      }\n    }\n\n    return report;\n  }\n}"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"CollectionCache.d.ts","sourceRoot":"","sources":["../../src/cache/CollectionCache.ts"],"names":[],"mappings":"AAAA;;GAEG;AAQH,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,GAAG,EAAE,MAAM,CAAC;IACZ,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,aAAa,CAAC,EAAE,MAAM,CAAC;CACxB;AAED,MAAM,WAAW,oBAAoB;IACnC,KAAK,EAAE,cAAc,EAAE,CAAC;IACxB,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AAED;;GAEG;AACH,qBAAa,eAAe;IAC1B,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAuB;gBAExC,OAAO,
|
|
1
|
+
{"version":3,"file":"CollectionCache.d.ts","sourceRoot":"","sources":["../../src/cache/CollectionCache.ts"],"names":[],"mappings":"AAAA;;GAEG;AAQH,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,GAAG,EAAE,MAAM,CAAC;IACZ,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,aAAa,CAAC,EAAE,MAAM,CAAC;CACxB;AAED,MAAM,WAAW,oBAAoB;IACnC,KAAK,EAAE,cAAc,EAAE,CAAC;IACxB,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AAED;;GAEG;AACH,qBAAa,eAAe;IAC1B,OAAO,CAAC,QAAQ,CAAS;IACzB,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAuB;gBAExC,OAAO,CAAC,EAAE,MAAM;IAc5B;;OAEG;YACW,cAAc;IAS5B;;OAEG;IACG,SAAS,IAAI,OAAO,CAAC,oBAAoB,GAAG,IAAI,CAAC;IAkCvD;;OAEG;IACG,SAAS,CAAC,KAAK,EAAE,cAAc,EAAE,EAAE,IAAI,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA0BtE;;OAEG;IACG,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,EAAE,CAAC;IAkB3D;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAO3B;;OAEG;IACG,cAAc,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,EAAE,CAAC;IASnE;;OAEG;IACG,YAAY,IAAI,OAAO,CAAC,OAAO,CAAC;IAKtC;;OAEG;IACG,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAWjC;;OAEG;IACG,aAAa,IAAI,OAAO,CAAC;QAAE,SAAS,EAAE,MAAM,CAAC;QAAC,QAAQ,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,OAAO,CAAA;KAAE,CAAC;CAY1F"}
|