@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,347 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Comprehensive Performance Monitoring System
|
|
3
|
+
* Tracks search times, memory usage, cache performance, and system metrics
|
|
4
|
+
*/
|
|
5
|
+
import { logger } from './logger.js';
|
|
6
|
+
import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
|
|
7
|
+
export class PerformanceMonitor {
|
|
8
|
+
static instance = null;
|
|
9
|
+
searchMetrics = [];
|
|
10
|
+
slowQueries = [];
|
|
11
|
+
memorySnapshots = [];
|
|
12
|
+
cacheMetrics = new Map();
|
|
13
|
+
// Configuration
|
|
14
|
+
maxMetricsHistory = 1000;
|
|
15
|
+
slowQueryThreshold = 100; // ms
|
|
16
|
+
memorySnapshotInterval = 30000; // 30 seconds
|
|
17
|
+
maxSlowQueries = 100;
|
|
18
|
+
// Timers and intervals
|
|
19
|
+
memoryMonitorInterval;
|
|
20
|
+
isMonitoring = false;
|
|
21
|
+
constructor() {
|
|
22
|
+
this.startMemoryMonitoring();
|
|
23
|
+
}
|
|
24
|
+
static getInstance() {
|
|
25
|
+
if (!this.instance) {
|
|
26
|
+
this.instance = new PerformanceMonitor();
|
|
27
|
+
}
|
|
28
|
+
return this.instance;
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Start performance monitoring
|
|
32
|
+
*/
|
|
33
|
+
startMonitoring() {
|
|
34
|
+
if (this.isMonitoring) {
|
|
35
|
+
return;
|
|
36
|
+
}
|
|
37
|
+
this.isMonitoring = true;
|
|
38
|
+
this.startMemoryMonitoring();
|
|
39
|
+
logger.info('Performance monitoring started', {
|
|
40
|
+
slowQueryThreshold: this.slowQueryThreshold,
|
|
41
|
+
maxMetricsHistory: this.maxMetricsHistory
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Stop performance monitoring
|
|
46
|
+
*/
|
|
47
|
+
stopMonitoring() {
|
|
48
|
+
this.isMonitoring = false;
|
|
49
|
+
if (this.memoryMonitorInterval) {
|
|
50
|
+
clearInterval(this.memoryMonitorInterval);
|
|
51
|
+
this.memoryMonitorInterval = undefined;
|
|
52
|
+
}
|
|
53
|
+
logger.info('Performance monitoring stopped');
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Record search performance metrics
|
|
57
|
+
*/
|
|
58
|
+
recordSearch(metrics) {
|
|
59
|
+
if (!this.isMonitoring) {
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
// Normalize query string to prevent Unicode-based attacks
|
|
63
|
+
const validationResult = UnicodeValidator.normalize(metrics.query);
|
|
64
|
+
const normalizedMetrics = {
|
|
65
|
+
...metrics,
|
|
66
|
+
query: validationResult.normalizedContent
|
|
67
|
+
};
|
|
68
|
+
this.searchMetrics.push(normalizedMetrics);
|
|
69
|
+
// Check if it's a slow query (use normalized metrics)
|
|
70
|
+
if (normalizedMetrics.duration > this.slowQueryThreshold) {
|
|
71
|
+
this.recordSlowQuery({
|
|
72
|
+
query: normalizedMetrics.query,
|
|
73
|
+
duration: normalizedMetrics.duration,
|
|
74
|
+
threshold: this.slowQueryThreshold,
|
|
75
|
+
sources: normalizedMetrics.sources,
|
|
76
|
+
resultCount: normalizedMetrics.resultCount,
|
|
77
|
+
memoryUsage: normalizedMetrics.memoryAfter,
|
|
78
|
+
timestamp: normalizedMetrics.timestamp
|
|
79
|
+
});
|
|
80
|
+
}
|
|
81
|
+
// Trim history if needed
|
|
82
|
+
if (this.searchMetrics.length > this.maxMetricsHistory) {
|
|
83
|
+
this.searchMetrics = this.searchMetrics.slice(-this.maxMetricsHistory);
|
|
84
|
+
}
|
|
85
|
+
// Log significant performance events (use normalized metrics)
|
|
86
|
+
if (normalizedMetrics.duration > this.slowQueryThreshold * 2) {
|
|
87
|
+
logger.warn('Very slow search detected', {
|
|
88
|
+
query: normalizedMetrics.query.substring(0, 50),
|
|
89
|
+
duration: normalizedMetrics.duration,
|
|
90
|
+
resultCount: normalizedMetrics.resultCount,
|
|
91
|
+
sources: normalizedMetrics.sources
|
|
92
|
+
});
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Record cache performance metrics
|
|
97
|
+
*/
|
|
98
|
+
recordCachePerformance(cacheName, stats) {
|
|
99
|
+
if (!this.isMonitoring) {
|
|
100
|
+
return;
|
|
101
|
+
}
|
|
102
|
+
// Normalize cache name to prevent Unicode-based attacks
|
|
103
|
+
const validationResult = UnicodeValidator.normalize(cacheName);
|
|
104
|
+
const normalizedCacheName = validationResult.normalizedContent;
|
|
105
|
+
this.cacheMetrics.set(normalizedCacheName, stats);
|
|
106
|
+
// Log cache performance warnings (use normalized cache name)
|
|
107
|
+
if (stats.hitRate < 0.5) {
|
|
108
|
+
logger.warn('Low cache hit rate detected', {
|
|
109
|
+
cache: normalizedCacheName,
|
|
110
|
+
hitRate: stats.hitRate,
|
|
111
|
+
totalOperations: stats.totalHits + stats.totalMisses
|
|
112
|
+
});
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Get comprehensive performance metrics
|
|
117
|
+
*/
|
|
118
|
+
getMetrics() {
|
|
119
|
+
return {
|
|
120
|
+
searchTimes: this.searchMetrics.map(m => m.duration),
|
|
121
|
+
memoryUsage: this.memorySnapshots.slice(-100), // Last 100 snapshots
|
|
122
|
+
cacheStats: this.aggregateCacheStats(),
|
|
123
|
+
systemStats: this.getSystemStats(),
|
|
124
|
+
timestamp: new Date()
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
/**
|
|
128
|
+
* Get search performance statistics
|
|
129
|
+
*/
|
|
130
|
+
getSearchStats() {
|
|
131
|
+
if (this.searchMetrics.length === 0) {
|
|
132
|
+
return {
|
|
133
|
+
totalSearches: 0,
|
|
134
|
+
averageTime: 0,
|
|
135
|
+
medianTime: 0,
|
|
136
|
+
p95Time: 0,
|
|
137
|
+
p99Time: 0,
|
|
138
|
+
slowQueries: 0,
|
|
139
|
+
cacheHitRate: 0
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
const times = this.searchMetrics.map(m => m.duration).sort((a, b) => a - b);
|
|
143
|
+
const cacheHits = this.searchMetrics.filter(m => m.cacheHit).length;
|
|
144
|
+
return {
|
|
145
|
+
totalSearches: this.searchMetrics.length,
|
|
146
|
+
averageTime: times.reduce((sum, time) => sum + time, 0) / times.length,
|
|
147
|
+
medianTime: times[Math.floor(times.length / 2)],
|
|
148
|
+
p95Time: times[Math.floor(times.length * 0.95)],
|
|
149
|
+
p99Time: times[Math.floor(times.length * 0.99)],
|
|
150
|
+
slowQueries: this.slowQueries.length,
|
|
151
|
+
cacheHitRate: cacheHits / this.searchMetrics.length
|
|
152
|
+
};
|
|
153
|
+
}
|
|
154
|
+
/**
|
|
155
|
+
* Get memory usage statistics
|
|
156
|
+
*/
|
|
157
|
+
getMemoryStats() {
|
|
158
|
+
if (this.memorySnapshots.length === 0) {
|
|
159
|
+
const current = this.takeMemorySnapshot();
|
|
160
|
+
return {
|
|
161
|
+
currentUsage: current,
|
|
162
|
+
peakUsage: current,
|
|
163
|
+
averageUsage: current,
|
|
164
|
+
growthRate: 0
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
const current = this.memorySnapshots[this.memorySnapshots.length - 1];
|
|
168
|
+
const peak = this.memorySnapshots.reduce((max, snapshot) => snapshot.heapUsed > max.heapUsed ? snapshot : max);
|
|
169
|
+
const totalHeap = this.memorySnapshots.reduce((sum, snapshot) => sum + snapshot.heapUsed, 0);
|
|
170
|
+
const totalRss = this.memorySnapshots.reduce((sum, snapshot) => sum + snapshot.rss, 0);
|
|
171
|
+
const average = {
|
|
172
|
+
heapUsed: totalHeap / this.memorySnapshots.length,
|
|
173
|
+
heapTotal: this.memorySnapshots.reduce((sum, s) => sum + s.heapTotal, 0) / this.memorySnapshots.length,
|
|
174
|
+
rss: totalRss / this.memorySnapshots.length,
|
|
175
|
+
external: this.memorySnapshots.reduce((sum, s) => sum + s.external, 0) / this.memorySnapshots.length,
|
|
176
|
+
timestamp: new Date()
|
|
177
|
+
};
|
|
178
|
+
// Calculate growth rate (MB per minute)
|
|
179
|
+
let growthRate = 0;
|
|
180
|
+
if (this.memorySnapshots.length > 1) {
|
|
181
|
+
const oldest = this.memorySnapshots[0];
|
|
182
|
+
const timeDiff = (current.timestamp.getTime() - oldest.timestamp.getTime()) / 60000; // minutes
|
|
183
|
+
const memoryDiff = (current.heapUsed - oldest.heapUsed) / (1024 * 1024); // MB
|
|
184
|
+
growthRate = timeDiff > 0 ? memoryDiff / timeDiff : 0;
|
|
185
|
+
}
|
|
186
|
+
return {
|
|
187
|
+
currentUsage: current,
|
|
188
|
+
peakUsage: peak,
|
|
189
|
+
averageUsage: average,
|
|
190
|
+
growthRate
|
|
191
|
+
};
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Get slow queries with analysis
|
|
195
|
+
*/
|
|
196
|
+
getSlowQueries(limit = 10) {
|
|
197
|
+
return this.slowQueries
|
|
198
|
+
.sort((a, b) => b.duration - a.duration)
|
|
199
|
+
.slice(0, limit);
|
|
200
|
+
}
|
|
201
|
+
/**
|
|
202
|
+
* Analyze performance trends
|
|
203
|
+
*/
|
|
204
|
+
analyzeTrends() {
|
|
205
|
+
const recommendations = [];
|
|
206
|
+
// Analyze search performance trend
|
|
207
|
+
let performanceTrend = 'stable';
|
|
208
|
+
if (this.searchMetrics.length > 10) {
|
|
209
|
+
const recent = this.searchMetrics.slice(-10).map(m => m.duration);
|
|
210
|
+
const older = this.searchMetrics.slice(-20, -10).map(m => m.duration);
|
|
211
|
+
if (recent.length > 0 && older.length > 0) {
|
|
212
|
+
const recentAvg = recent.reduce((sum, t) => sum + t, 0) / recent.length;
|
|
213
|
+
const olderAvg = older.reduce((sum, t) => sum + t, 0) / older.length;
|
|
214
|
+
if (recentAvg > olderAvg * 1.2) {
|
|
215
|
+
performanceTrend = 'degrading';
|
|
216
|
+
recommendations.push('Search performance is degrading. Consider cache optimization or index rebuilding.');
|
|
217
|
+
}
|
|
218
|
+
else if (recentAvg < olderAvg * 0.8) {
|
|
219
|
+
performanceTrend = 'improving';
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
// Analyze memory trend
|
|
224
|
+
const memoryStats = this.getMemoryStats();
|
|
225
|
+
let memoryTrend = 'stable';
|
|
226
|
+
if (memoryStats.growthRate > 1) { // Growing by more than 1MB/minute
|
|
227
|
+
memoryTrend = 'growing';
|
|
228
|
+
recommendations.push('Memory usage is growing rapidly. Consider cache cleanup or memory limits.');
|
|
229
|
+
}
|
|
230
|
+
else if (memoryStats.growthRate < -1) {
|
|
231
|
+
memoryTrend = 'shrinking';
|
|
232
|
+
}
|
|
233
|
+
// Cache performance recommendations
|
|
234
|
+
const cacheStats = this.aggregateCacheStats();
|
|
235
|
+
if (cacheStats.hitRate < 0.6) {
|
|
236
|
+
recommendations.push('Cache hit rate is low. Consider adjusting cache size or TTL settings.');
|
|
237
|
+
}
|
|
238
|
+
// Slow query recommendations
|
|
239
|
+
if (this.slowQueries.length > 10) {
|
|
240
|
+
recommendations.push('Multiple slow queries detected. Consider query optimization or increased caching.');
|
|
241
|
+
}
|
|
242
|
+
return {
|
|
243
|
+
performanceTrend,
|
|
244
|
+
memoryTrend,
|
|
245
|
+
recommendations
|
|
246
|
+
};
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Reset all performance metrics
|
|
250
|
+
*/
|
|
251
|
+
reset() {
|
|
252
|
+
this.searchMetrics = [];
|
|
253
|
+
this.slowQueries = [];
|
|
254
|
+
this.memorySnapshots = [];
|
|
255
|
+
this.cacheMetrics.clear();
|
|
256
|
+
logger.info('Performance metrics reset');
|
|
257
|
+
}
|
|
258
|
+
/**
|
|
259
|
+
* Export metrics for external analysis
|
|
260
|
+
*/
|
|
261
|
+
exportMetrics() {
|
|
262
|
+
const data = {
|
|
263
|
+
searchMetrics: this.searchMetrics,
|
|
264
|
+
slowQueries: this.slowQueries,
|
|
265
|
+
memorySnapshots: this.memorySnapshots,
|
|
266
|
+
cacheMetrics: Object.fromEntries(this.cacheMetrics),
|
|
267
|
+
exportTimestamp: new Date().toISOString()
|
|
268
|
+
};
|
|
269
|
+
return JSON.stringify(data, null, 2);
|
|
270
|
+
}
|
|
271
|
+
// Private methods
|
|
272
|
+
recordSlowQuery(query) {
|
|
273
|
+
this.slowQueries.push(query);
|
|
274
|
+
// Trim slow queries history
|
|
275
|
+
if (this.slowQueries.length > this.maxSlowQueries) {
|
|
276
|
+
this.slowQueries = this.slowQueries.slice(-this.maxSlowQueries);
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
startMemoryMonitoring() {
|
|
280
|
+
if (this.memoryMonitorInterval) {
|
|
281
|
+
clearInterval(this.memoryMonitorInterval);
|
|
282
|
+
}
|
|
283
|
+
this.memoryMonitorInterval = setInterval(() => {
|
|
284
|
+
if (this.isMonitoring) {
|
|
285
|
+
const snapshot = this.takeMemorySnapshot();
|
|
286
|
+
this.memorySnapshots.push(snapshot);
|
|
287
|
+
// Trim memory snapshots (keep last 200)
|
|
288
|
+
if (this.memorySnapshots.length > 200) {
|
|
289
|
+
this.memorySnapshots = this.memorySnapshots.slice(-200);
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
}, this.memorySnapshotInterval);
|
|
293
|
+
}
|
|
294
|
+
takeMemorySnapshot() {
|
|
295
|
+
const memUsage = process.memoryUsage();
|
|
296
|
+
return {
|
|
297
|
+
heapUsed: memUsage.heapUsed,
|
|
298
|
+
heapTotal: memUsage.heapTotal,
|
|
299
|
+
rss: memUsage.rss,
|
|
300
|
+
external: memUsage.external,
|
|
301
|
+
timestamp: new Date()
|
|
302
|
+
};
|
|
303
|
+
}
|
|
304
|
+
aggregateCacheStats() {
|
|
305
|
+
if (this.cacheMetrics.size === 0) {
|
|
306
|
+
return {
|
|
307
|
+
hitRate: 0,
|
|
308
|
+
avgHitTime: 0,
|
|
309
|
+
avgMissTime: 0,
|
|
310
|
+
totalHits: 0,
|
|
311
|
+
totalMisses: 0,
|
|
312
|
+
evictions: 0
|
|
313
|
+
};
|
|
314
|
+
}
|
|
315
|
+
let totalHits = 0;
|
|
316
|
+
let totalMisses = 0;
|
|
317
|
+
let totalEvictions = 0;
|
|
318
|
+
let weightedHitTime = 0;
|
|
319
|
+
let weightedMissTime = 0;
|
|
320
|
+
for (const stats of this.cacheMetrics.values()) {
|
|
321
|
+
totalHits += stats.totalHits;
|
|
322
|
+
totalMisses += stats.totalMisses;
|
|
323
|
+
totalEvictions += stats.evictions;
|
|
324
|
+
weightedHitTime += stats.avgHitTime * stats.totalHits;
|
|
325
|
+
weightedMissTime += stats.avgMissTime * stats.totalMisses;
|
|
326
|
+
}
|
|
327
|
+
return {
|
|
328
|
+
hitRate: totalHits + totalMisses > 0 ? totalHits / (totalHits + totalMisses) : 0,
|
|
329
|
+
avgHitTime: totalHits > 0 ? weightedHitTime / totalHits : 0,
|
|
330
|
+
avgMissTime: totalMisses > 0 ? weightedMissTime / totalMisses : 0,
|
|
331
|
+
totalHits,
|
|
332
|
+
totalMisses,
|
|
333
|
+
evictions: totalEvictions
|
|
334
|
+
};
|
|
335
|
+
}
|
|
336
|
+
getSystemStats() {
|
|
337
|
+
const os = require('os');
|
|
338
|
+
return {
|
|
339
|
+
cpuUsage: process.cpuUsage().user / 1000000, // Convert to seconds
|
|
340
|
+
loadAverage: os.loadavg(),
|
|
341
|
+
freeMemory: os.freemem(),
|
|
342
|
+
totalMemory: os.totalmem(),
|
|
343
|
+
uptime: process.uptime()
|
|
344
|
+
};
|
|
345
|
+
}
|
|
346
|
+
}
|
|
347
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PerformanceMonitor.js","sourceRoot":"","sources":["../../src/utils/PerformanceMonitor.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Comprehensive Performance Monitoring System\n * Tracks search times, memory usage, cache performance, and system metrics\n */\n\nimport { logger } from './logger.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\n\nexport interface PerformanceMetrics {\n  searchTimes: number[];\n  memoryUsage: MemoryUsage[];\n  cacheStats: CachePerformance;\n  systemStats: SystemStats;\n  timestamp: Date;\n}\n\nexport interface MemoryUsage {\n  heapUsed: number;\n  heapTotal: number;\n  rss: number;\n  external: number;\n  timestamp: Date;\n}\n\nexport interface CachePerformance {\n  hitRate: number;\n  avgHitTime: number;\n  avgMissTime: number;\n  totalHits: number;\n  totalMisses: number;\n  evictions: number;\n}\n\nexport interface SystemStats {\n  cpuUsage: number;\n  loadAverage: number[];\n  freeMemory: number;\n  totalMemory: number;\n  uptime: number;\n}\n\nexport interface SearchMetrics {\n  query: string;\n  duration: number;\n  resultCount: number;\n  sources: string[];\n  cacheHit: boolean;\n  memoryBefore: number;\n  memoryAfter: number;\n  timestamp: Date;\n}\n\nexport interface SlowQuery {\n  query: string;\n  duration: number;\n  threshold: number;\n  sources: string[];\n  resultCount: number;\n  memoryUsage: number;\n  timestamp: Date;\n}\n\nexport class PerformanceMonitor {\n  private static instance: PerformanceMonitor | null = null;\n\n  private searchMetrics: SearchMetrics[] = [];\n  private slowQueries: SlowQuery[] = [];\n  private memorySnapshots: MemoryUsage[] = [];\n  private cacheMetrics: Map<string, CachePerformance> = new Map();\n\n  // Configuration\n  private readonly maxMetricsHistory = 1000;\n  private readonly slowQueryThreshold = 100; // ms\n  private readonly memorySnapshotInterval = 30000; // 30 seconds\n  private readonly maxSlowQueries = 100;\n\n  // Timers and intervals\n  private memoryMonitorInterval?: NodeJS.Timeout;\n  private isMonitoring = false;\n\n  private constructor() {\n    this.startMemoryMonitoring();\n  }\n\n  public static getInstance(): PerformanceMonitor {\n    if (!this.instance) {\n      this.instance = new PerformanceMonitor();\n    }\n    return this.instance;\n  }\n\n  /**\n   * Start performance monitoring\n   */\n  startMonitoring(): void {\n    if (this.isMonitoring) {\n      return;\n    }\n\n    this.isMonitoring = true;\n    this.startMemoryMonitoring();\n    \n    logger.info('Performance monitoring started', {\n      slowQueryThreshold: this.slowQueryThreshold,\n      maxMetricsHistory: this.maxMetricsHistory\n    });\n  }\n\n  /**\n   * Stop performance monitoring\n   */\n  stopMonitoring(): void {\n    this.isMonitoring = false;\n    \n    if (this.memoryMonitorInterval) {\n      clearInterval(this.memoryMonitorInterval);\n      this.memoryMonitorInterval = undefined;\n    }\n\n    logger.info('Performance monitoring stopped');\n  }\n\n  /**\n   * Record search performance metrics\n   */\n  recordSearch(metrics: SearchMetrics): void {\n    if (!this.isMonitoring) {\n      return;\n    }\n\n    // Normalize query string to prevent Unicode-based attacks\n    const validationResult = UnicodeValidator.normalize(metrics.query);\n    const normalizedMetrics = {\n      ...metrics,\n      query: validationResult.normalizedContent\n    };\n\n    this.searchMetrics.push(normalizedMetrics);\n\n    // Check if it's a slow query (use normalized metrics)\n    if (normalizedMetrics.duration > this.slowQueryThreshold) {\n      this.recordSlowQuery({\n        query: normalizedMetrics.query,\n        duration: normalizedMetrics.duration,\n        threshold: this.slowQueryThreshold,\n        sources: normalizedMetrics.sources,\n        resultCount: normalizedMetrics.resultCount,\n        memoryUsage: normalizedMetrics.memoryAfter,\n        timestamp: normalizedMetrics.timestamp\n      });\n    }\n\n    // Trim history if needed\n    if (this.searchMetrics.length > this.maxMetricsHistory) {\n      this.searchMetrics = this.searchMetrics.slice(-this.maxMetricsHistory);\n    }\n\n    // Log significant performance events (use normalized metrics)\n    if (normalizedMetrics.duration > this.slowQueryThreshold * 2) {\n      logger.warn('Very slow search detected', {\n        query: normalizedMetrics.query.substring(0, 50),\n        duration: normalizedMetrics.duration,\n        resultCount: normalizedMetrics.resultCount,\n        sources: normalizedMetrics.sources\n      });\n    }\n  }\n\n  /**\n   * Record cache performance metrics\n   */\n  recordCachePerformance(cacheName: string, stats: CachePerformance): void {\n    if (!this.isMonitoring) {\n      return;\n    }\n\n    // Normalize cache name to prevent Unicode-based attacks\n    const validationResult = UnicodeValidator.normalize(cacheName);\n    const normalizedCacheName = validationResult.normalizedContent;\n\n    this.cacheMetrics.set(normalizedCacheName, stats);\n\n    // Log cache performance warnings (use normalized cache name)\n    if (stats.hitRate < 0.5) {\n      logger.warn('Low cache hit rate detected', {\n        cache: normalizedCacheName,\n        hitRate: stats.hitRate,\n        totalOperations: stats.totalHits + stats.totalMisses\n      });\n    }\n  }\n\n  /**\n   * Get comprehensive performance metrics\n   */\n  getMetrics(): PerformanceMetrics {\n    return {\n      searchTimes: this.searchMetrics.map(m => m.duration),\n      memoryUsage: this.memorySnapshots.slice(-100), // Last 100 snapshots\n      cacheStats: this.aggregateCacheStats(),\n      systemStats: this.getSystemStats(),\n      timestamp: new Date()\n    };\n  }\n\n  /**\n   * Get search performance statistics\n   */\n  getSearchStats(): {\n    totalSearches: number;\n    averageTime: number;\n    medianTime: number;\n    p95Time: number;\n    p99Time: number;\n    slowQueries: number;\n    cacheHitRate: number;\n  } {\n    if (this.searchMetrics.length === 0) {\n      return {\n        totalSearches: 0,\n        averageTime: 0,\n        medianTime: 0,\n        p95Time: 0,\n        p99Time: 0,\n        slowQueries: 0,\n        cacheHitRate: 0\n      };\n    }\n\n    const times = this.searchMetrics.map(m => m.duration).sort((a, b) => a - b);\n    const cacheHits = this.searchMetrics.filter(m => m.cacheHit).length;\n\n    return {\n      totalSearches: this.searchMetrics.length,\n      averageTime: times.reduce((sum, time) => sum + time, 0) / times.length,\n      medianTime: times[Math.floor(times.length / 2)],\n      p95Time: times[Math.floor(times.length * 0.95)],\n      p99Time: times[Math.floor(times.length * 0.99)],\n      slowQueries: this.slowQueries.length,\n      cacheHitRate: cacheHits / this.searchMetrics.length\n    };\n  }\n\n  /**\n   * Get memory usage statistics\n   */\n  getMemoryStats(): {\n    currentUsage: MemoryUsage;\n    peakUsage: MemoryUsage;\n    averageUsage: MemoryUsage;\n    growthRate: number; // MB per minute\n  } {\n    if (this.memorySnapshots.length === 0) {\n      const current = this.takeMemorySnapshot();\n      return {\n        currentUsage: current,\n        peakUsage: current,\n        averageUsage: current,\n        growthRate: 0\n      };\n    }\n\n    const current = this.memorySnapshots[this.memorySnapshots.length - 1];\n    const peak = this.memorySnapshots.reduce((max, snapshot) => \n      snapshot.heapUsed > max.heapUsed ? snapshot : max\n    );\n\n    const totalHeap = this.memorySnapshots.reduce((sum, snapshot) => sum + snapshot.heapUsed, 0);\n    const totalRss = this.memorySnapshots.reduce((sum, snapshot) => sum + snapshot.rss, 0);\n    const average: MemoryUsage = {\n      heapUsed: totalHeap / this.memorySnapshots.length,\n      heapTotal: this.memorySnapshots.reduce((sum, s) => sum + s.heapTotal, 0) / this.memorySnapshots.length,\n      rss: totalRss / this.memorySnapshots.length,\n      external: this.memorySnapshots.reduce((sum, s) => sum + s.external, 0) / this.memorySnapshots.length,\n      timestamp: new Date()\n    };\n\n    // Calculate growth rate (MB per minute)\n    let growthRate = 0;\n    if (this.memorySnapshots.length > 1) {\n      const oldest = this.memorySnapshots[0];\n      const timeDiff = (current.timestamp.getTime() - oldest.timestamp.getTime()) / 60000; // minutes\n      const memoryDiff = (current.heapUsed - oldest.heapUsed) / (1024 * 1024); // MB\n      growthRate = timeDiff > 0 ? memoryDiff / timeDiff : 0;\n    }\n\n    return {\n      currentUsage: current,\n      peakUsage: peak,\n      averageUsage: average,\n      growthRate\n    };\n  }\n\n  /**\n   * Get slow queries with analysis\n   */\n  getSlowQueries(limit: number = 10): SlowQuery[] {\n    return this.slowQueries\n      .sort((a, b) => b.duration - a.duration)\n      .slice(0, limit);\n  }\n\n  /**\n   * Analyze performance trends\n   */\n  analyzeTrends(): {\n    performanceTrend: 'improving' | 'degrading' | 'stable';\n    memoryTrend: 'growing' | 'shrinking' | 'stable';\n    recommendations: string[];\n  } {\n    const recommendations: string[] = [];\n    \n    // Analyze search performance trend\n    let performanceTrend: 'improving' | 'degrading' | 'stable' = 'stable';\n    if (this.searchMetrics.length > 10) {\n      const recent = this.searchMetrics.slice(-10).map(m => m.duration);\n      const older = this.searchMetrics.slice(-20, -10).map(m => m.duration);\n      \n      if (recent.length > 0 && older.length > 0) {\n        const recentAvg = recent.reduce((sum, t) => sum + t, 0) / recent.length;\n        const olderAvg = older.reduce((sum, t) => sum + t, 0) / older.length;\n        \n        if (recentAvg > olderAvg * 1.2) {\n          performanceTrend = 'degrading';\n          recommendations.push('Search performance is degrading. Consider cache optimization or index rebuilding.');\n        } else if (recentAvg < olderAvg * 0.8) {\n          performanceTrend = 'improving';\n        }\n      }\n    }\n\n    // Analyze memory trend\n    const memoryStats = this.getMemoryStats();\n    let memoryTrend: 'growing' | 'shrinking' | 'stable' = 'stable';\n    \n    if (memoryStats.growthRate > 1) { // Growing by more than 1MB/minute\n      memoryTrend = 'growing';\n      recommendations.push('Memory usage is growing rapidly. Consider cache cleanup or memory limits.');\n    } else if (memoryStats.growthRate < -1) {\n      memoryTrend = 'shrinking';\n    }\n\n    // Cache performance recommendations\n    const cacheStats = this.aggregateCacheStats();\n    if (cacheStats.hitRate < 0.6) {\n      recommendations.push('Cache hit rate is low. Consider adjusting cache size or TTL settings.');\n    }\n\n    // Slow query recommendations\n    if (this.slowQueries.length > 10) {\n      recommendations.push('Multiple slow queries detected. Consider query optimization or increased caching.');\n    }\n\n    return {\n      performanceTrend,\n      memoryTrend,\n      recommendations\n    };\n  }\n\n  /**\n   * Reset all performance metrics\n   */\n  reset(): void {\n    this.searchMetrics = [];\n    this.slowQueries = [];\n    this.memorySnapshots = [];\n    this.cacheMetrics.clear();\n    \n    logger.info('Performance metrics reset');\n  }\n\n  /**\n   * Export metrics for external analysis\n   */\n  exportMetrics(): string {\n    const data = {\n      searchMetrics: this.searchMetrics,\n      slowQueries: this.slowQueries,\n      memorySnapshots: this.memorySnapshots,\n      cacheMetrics: Object.fromEntries(this.cacheMetrics),\n      exportTimestamp: new Date().toISOString()\n    };\n\n    return JSON.stringify(data, null, 2);\n  }\n\n  // Private methods\n\n  private recordSlowQuery(query: SlowQuery): void {\n    this.slowQueries.push(query);\n\n    // Trim slow queries history\n    if (this.slowQueries.length > this.maxSlowQueries) {\n      this.slowQueries = this.slowQueries.slice(-this.maxSlowQueries);\n    }\n  }\n\n  private startMemoryMonitoring(): void {\n    if (this.memoryMonitorInterval) {\n      clearInterval(this.memoryMonitorInterval);\n    }\n\n    this.memoryMonitorInterval = setInterval(() => {\n      if (this.isMonitoring) {\n        const snapshot = this.takeMemorySnapshot();\n        this.memorySnapshots.push(snapshot);\n\n        // Trim memory snapshots (keep last 200)\n        if (this.memorySnapshots.length > 200) {\n          this.memorySnapshots = this.memorySnapshots.slice(-200);\n        }\n      }\n    }, this.memorySnapshotInterval);\n  }\n\n  private takeMemorySnapshot(): MemoryUsage {\n    const memUsage = process.memoryUsage();\n    return {\n      heapUsed: memUsage.heapUsed,\n      heapTotal: memUsage.heapTotal,\n      rss: memUsage.rss,\n      external: memUsage.external,\n      timestamp: new Date()\n    };\n  }\n\n  private aggregateCacheStats(): CachePerformance {\n    if (this.cacheMetrics.size === 0) {\n      return {\n        hitRate: 0,\n        avgHitTime: 0,\n        avgMissTime: 0,\n        totalHits: 0,\n        totalMisses: 0,\n        evictions: 0\n      };\n    }\n\n    let totalHits = 0;\n    let totalMisses = 0;\n    let totalEvictions = 0;\n    let weightedHitTime = 0;\n    let weightedMissTime = 0;\n\n    for (const stats of this.cacheMetrics.values()) {\n      totalHits += stats.totalHits;\n      totalMisses += stats.totalMisses;\n      totalEvictions += stats.evictions;\n      weightedHitTime += stats.avgHitTime * stats.totalHits;\n      weightedMissTime += stats.avgMissTime * stats.totalMisses;\n    }\n\n    return {\n      hitRate: totalHits + totalMisses > 0 ? totalHits / (totalHits + totalMisses) : 0,\n      avgHitTime: totalHits > 0 ? weightedHitTime / totalHits : 0,\n      avgMissTime: totalMisses > 0 ? weightedMissTime / totalMisses : 0,\n      totalHits,\n      totalMisses,\n      evictions: totalEvictions\n    };\n  }\n\n  private getSystemStats(): SystemStats {\n    const os = require('os');\n    \n    return {\n      cpuUsage: process.cpuUsage().user / 1000000, // Convert to seconds\n      loadAverage: os.loadavg(),\n      freeMemory: os.freemem(),\n      totalMemory: os.totalmem(),\n      uptime: process.uptime()\n    };\n  }\n}"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"RateLimiter.d.ts","sourceRoot":"","sources":["../../src/utils/RateLimiter.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,MAAM,WAAW,iBAAiB;IAChC,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,EAAE,MAAM,CAAC;IACjB,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,eAAe;IAC9B,OAAO,EAAE,OAAO,CAAC;IACjB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,eAAe,EAAE,MAAM,CAAC;IACxB,SAAS,EAAE,IAAI,CAAC;CACjB;AAED,qBAAa,WAAW;IACtB,OAAO,CAAC,MAAM,CAAS;IACvB,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAS;IACnC,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAS;IACpC,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAS;gBAEtB,MAAM,EAAE,iBAAiB;IAsBrC;;;OAGG;IACH,UAAU,IAAI,eAAe;IA0C7B;;;OAGG;IACH,YAAY,IAAI,IAAI;IAUpB;;OAEG;IACH,SAAS,IAAI,eAAe;IAW5B;;;OAGG;IACH,KAAK,IAAI,IAAI;IAMb;;OAEG;IACH,OAAO,CAAC,YAAY;IAQpB;;OAEG;IACH,OAAO,CAAC,YAAY;IAOpB;;OAEG;IACH,QAAQ,IAAI,MAAM;CAKnB;AAED;;GAEG;AACH,qBAAa,kBAAkB;IAC7B;;OAEG;IACH,MAAM,CAAC,mBAAmB,IAAI,WAAW;IAQzC;;;OAGG;IACH,MAAM,CAAC,wBAAwB,IAAI,WAAW;IAQ9C;;;OAGG;IACH,MAAM,CAAC,mBAAmB,IAAI,WAAW;CAO1C"}
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* RateLimiter - Implements rate limiting for API calls to prevent abuse
|
|
3
|
+
*
|
|
4
|
+
* Features:
|
|
5
|
+
* - Token bucket algorithm for flexible rate limiting
|
|
6
|
+
* - Configurable limits per time window
|
|
7
|
+
* - Memory-efficient implementation
|
|
8
|
+
* - Thread-safe for concurrent requests
|
|
9
|
+
*/
|
|
10
|
+
export class RateLimiter {
|
|
11
|
+
tokens;
|
|
12
|
+
lastRefill;
|
|
13
|
+
lastRequest;
|
|
14
|
+
maxTokens;
|
|
15
|
+
refillRate;
|
|
16
|
+
minDelay;
|
|
17
|
+
constructor(config) {
|
|
18
|
+
if (config.maxRequests <= 0) {
|
|
19
|
+
throw new Error('maxRequests must be positive');
|
|
20
|
+
}
|
|
21
|
+
if (config.windowMs <= 0) {
|
|
22
|
+
throw new Error('windowMs must be positive');
|
|
23
|
+
}
|
|
24
|
+
this.maxTokens = config.maxRequests;
|
|
25
|
+
this.tokens = this.maxTokens;
|
|
26
|
+
this.refillRate = this.maxTokens / config.windowMs;
|
|
27
|
+
// Validate refill rate to prevent division by zero
|
|
28
|
+
if (this.refillRate <= 0 || !isFinite(this.refillRate)) {
|
|
29
|
+
throw new Error('Invalid configuration: refill rate must be positive and finite');
|
|
30
|
+
}
|
|
31
|
+
this.lastRefill = Date.now();
|
|
32
|
+
this.lastRequest = 0;
|
|
33
|
+
this.minDelay = config.minDelayMs || 0;
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Check if a request is allowed under the rate limit
|
|
37
|
+
* @returns Status object indicating if request is allowed
|
|
38
|
+
*/
|
|
39
|
+
checkLimit() {
|
|
40
|
+
const now = Date.now();
|
|
41
|
+
// Refill tokens based on time elapsed
|
|
42
|
+
this.refillTokens(now);
|
|
43
|
+
// Check minimum delay between requests
|
|
44
|
+
if (this.minDelay > 0 && this.lastRequest > 0) {
|
|
45
|
+
const timeSinceLastRequest = now - this.lastRequest;
|
|
46
|
+
if (timeSinceLastRequest < this.minDelay) {
|
|
47
|
+
const retryAfterMs = this.minDelay - timeSinceLastRequest;
|
|
48
|
+
return {
|
|
49
|
+
allowed: false,
|
|
50
|
+
retryAfterMs,
|
|
51
|
+
remainingTokens: Math.floor(this.tokens),
|
|
52
|
+
resetTime: new Date(now + retryAfterMs)
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
// Check if we have tokens available
|
|
57
|
+
if (this.tokens < 1) {
|
|
58
|
+
// Calculate when the next token will be available
|
|
59
|
+
const tokensNeeded = 1 - this.tokens;
|
|
60
|
+
const msUntilNextToken = tokensNeeded / this.refillRate;
|
|
61
|
+
return {
|
|
62
|
+
allowed: false,
|
|
63
|
+
retryAfterMs: Math.ceil(msUntilNextToken),
|
|
64
|
+
remainingTokens: 0,
|
|
65
|
+
resetTime: new Date(now + msUntilNextToken)
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
// Request is allowed
|
|
69
|
+
return {
|
|
70
|
+
allowed: true,
|
|
71
|
+
remainingTokens: Math.floor(this.tokens),
|
|
72
|
+
resetTime: this.getResetTime()
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Consume a token for an allowed request
|
|
77
|
+
* Should be called after checkLimit() returns allowed: true
|
|
78
|
+
*/
|
|
79
|
+
consumeToken() {
|
|
80
|
+
const now = Date.now();
|
|
81
|
+
this.refillTokens(now);
|
|
82
|
+
if (this.tokens >= 1) {
|
|
83
|
+
this.tokens -= 1;
|
|
84
|
+
this.lastRequest = now;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Get current rate limit status without consuming a token
|
|
89
|
+
*/
|
|
90
|
+
getStatus() {
|
|
91
|
+
const now = Date.now();
|
|
92
|
+
this.refillTokens(now);
|
|
93
|
+
return {
|
|
94
|
+
allowed: this.tokens >= 1,
|
|
95
|
+
remainingTokens: Math.floor(this.tokens),
|
|
96
|
+
resetTime: this.getResetTime()
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Reset the rate limiter to full capacity
|
|
101
|
+
* Useful for testing or manual intervention
|
|
102
|
+
*/
|
|
103
|
+
reset() {
|
|
104
|
+
this.tokens = this.maxTokens;
|
|
105
|
+
this.lastRefill = Date.now();
|
|
106
|
+
this.lastRequest = 0;
|
|
107
|
+
}
|
|
108
|
+
/**
|
|
109
|
+
* Refill tokens based on time elapsed
|
|
110
|
+
*/
|
|
111
|
+
refillTokens(now) {
|
|
112
|
+
const timeSinceLastRefill = now - this.lastRefill;
|
|
113
|
+
const tokensToAdd = timeSinceLastRefill * this.refillRate;
|
|
114
|
+
this.tokens = Math.min(this.maxTokens, this.tokens + tokensToAdd);
|
|
115
|
+
this.lastRefill = now;
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Calculate when the rate limit window will reset
|
|
119
|
+
*/
|
|
120
|
+
getResetTime() {
|
|
121
|
+
const now = Date.now();
|
|
122
|
+
const tokensToFull = this.maxTokens - this.tokens;
|
|
123
|
+
const msUntilFull = tokensToFull / this.refillRate;
|
|
124
|
+
return new Date(now + msUntilFull);
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Get human-readable rate limit information
|
|
128
|
+
*/
|
|
129
|
+
toString() {
|
|
130
|
+
const status = this.getStatus();
|
|
131
|
+
return `RateLimit: ${status.remainingTokens}/${this.maxTokens} tokens, ` +
|
|
132
|
+
`resets at ${status.resetTime.toISOString()}`;
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
/**
|
|
136
|
+
* Factory function to create common rate limiters
|
|
137
|
+
*/
|
|
138
|
+
export class RateLimiterFactory {
|
|
139
|
+
/**
|
|
140
|
+
* GitHub API rate limiter (60 requests per hour for unauthenticated)
|
|
141
|
+
*/
|
|
142
|
+
static createGitHubLimiter() {
|
|
143
|
+
return new RateLimiter({
|
|
144
|
+
maxRequests: 60,
|
|
145
|
+
windowMs: 60 * 60 * 1000, // 1 hour
|
|
146
|
+
minDelayMs: 1000 // 1 second minimum between requests
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Conservative rate limiter for update checks
|
|
151
|
+
* Allows 10 checks per hour with 30 second minimum delay
|
|
152
|
+
*/
|
|
153
|
+
static createUpdateCheckLimiter() {
|
|
154
|
+
return new RateLimiter({
|
|
155
|
+
maxRequests: 10,
|
|
156
|
+
windowMs: 60 * 60 * 1000, // 1 hour
|
|
157
|
+
minDelayMs: 30 * 1000 // 30 seconds between checks
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
/**
|
|
161
|
+
* Strict rate limiter for sensitive operations
|
|
162
|
+
* Allows 5 requests per hour with 1 minute minimum delay
|
|
163
|
+
*/
|
|
164
|
+
static createStrictLimiter() {
|
|
165
|
+
return new RateLimiter({
|
|
166
|
+
maxRequests: 5,
|
|
167
|
+
windowMs: 60 * 60 * 1000, // 1 hour
|
|
168
|
+
minDelayMs: 60 * 1000 // 1 minute between requests
|
|
169
|
+
});
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RateLimiter.js","sourceRoot":"","sources":["../../src/utils/RateLimiter.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * RateLimiter - Implements rate limiting for API calls to prevent abuse\n * \n * Features:\n * - Token bucket algorithm for flexible rate limiting\n * - Configurable limits per time window\n * - Memory-efficient implementation\n * - Thread-safe for concurrent requests\n */\n\nexport interface RateLimiterConfig {\n  maxRequests: number;      // Maximum requests allowed\n  windowMs: number;         // Time window in milliseconds\n  minDelayMs?: number;      // Minimum delay between requests (optional)\n}\n\nexport interface RateLimitStatus {\n  allowed: boolean;\n  retryAfterMs?: number;\n  remainingTokens: number;\n  resetTime: Date;\n}\n\nexport class RateLimiter {\n  private tokens: number;\n  private lastRefill: number;\n  private lastRequest: number;\n  private readonly maxTokens: number;\n  private readonly refillRate: number;\n  private readonly minDelay: number;\n\n  constructor(config: RateLimiterConfig) {\n    if (config.maxRequests <= 0) {\n      throw new Error('maxRequests must be positive');\n    }\n    if (config.windowMs <= 0) {\n      throw new Error('windowMs must be positive');\n    }\n\n    this.maxTokens = config.maxRequests;\n    this.tokens = this.maxTokens;\n    this.refillRate = this.maxTokens / config.windowMs;\n    \n    // Validate refill rate to prevent division by zero\n    if (this.refillRate <= 0 || !isFinite(this.refillRate)) {\n      throw new Error('Invalid configuration: refill rate must be positive and finite');\n    }\n    \n    this.lastRefill = Date.now();\n    this.lastRequest = 0;\n    this.minDelay = config.minDelayMs || 0;\n  }\n\n  /**\n   * Check if a request is allowed under the rate limit\n   * @returns Status object indicating if request is allowed\n   */\n  checkLimit(): RateLimitStatus {\n    const now = Date.now();\n    \n    // Refill tokens based on time elapsed\n    this.refillTokens(now);\n\n    // Check minimum delay between requests\n    if (this.minDelay > 0 && this.lastRequest > 0) {\n      const timeSinceLastRequest = now - this.lastRequest;\n      if (timeSinceLastRequest < this.minDelay) {\n        const retryAfterMs = this.minDelay - timeSinceLastRequest;\n        return {\n          allowed: false,\n          retryAfterMs,\n          remainingTokens: Math.floor(this.tokens),\n          resetTime: new Date(now + retryAfterMs)\n        };\n      }\n    }\n\n    // Check if we have tokens available\n    if (this.tokens < 1) {\n      // Calculate when the next token will be available\n      const tokensNeeded = 1 - this.tokens;\n      const msUntilNextToken = tokensNeeded / this.refillRate;\n      \n      return {\n        allowed: false,\n        retryAfterMs: Math.ceil(msUntilNextToken),\n        remainingTokens: 0,\n        resetTime: new Date(now + msUntilNextToken)\n      };\n    }\n\n    // Request is allowed\n    return {\n      allowed: true,\n      remainingTokens: Math.floor(this.tokens),\n      resetTime: this.getResetTime()\n    };\n  }\n\n  /**\n   * Consume a token for an allowed request\n   * Should be called after checkLimit() returns allowed: true\n   */\n  consumeToken(): void {\n    const now = Date.now();\n    this.refillTokens(now);\n    \n    if (this.tokens >= 1) {\n      this.tokens -= 1;\n      this.lastRequest = now;\n    }\n  }\n\n  /**\n   * Get current rate limit status without consuming a token\n   */\n  getStatus(): RateLimitStatus {\n    const now = Date.now();\n    this.refillTokens(now);\n\n    return {\n      allowed: this.tokens >= 1,\n      remainingTokens: Math.floor(this.tokens),\n      resetTime: this.getResetTime()\n    };\n  }\n\n  /**\n   * Reset the rate limiter to full capacity\n   * Useful for testing or manual intervention\n   */\n  reset(): void {\n    this.tokens = this.maxTokens;\n    this.lastRefill = Date.now();\n    this.lastRequest = 0;\n  }\n\n  /**\n   * Refill tokens based on time elapsed\n   */\n  private refillTokens(now: number): void {\n    const timeSinceLastRefill = now - this.lastRefill;\n    const tokensToAdd = timeSinceLastRefill * this.refillRate;\n    \n    this.tokens = Math.min(this.maxTokens, this.tokens + tokensToAdd);\n    this.lastRefill = now;\n  }\n\n  /**\n   * Calculate when the rate limit window will reset\n   */\n  private getResetTime(): Date {\n    const now = Date.now();\n    const tokensToFull = this.maxTokens - this.tokens;\n    const msUntilFull = tokensToFull / this.refillRate;\n    return new Date(now + msUntilFull);\n  }\n\n  /**\n   * Get human-readable rate limit information\n   */\n  toString(): string {\n    const status = this.getStatus();\n    return `RateLimit: ${status.remainingTokens}/${this.maxTokens} tokens, ` +\n           `resets at ${status.resetTime.toISOString()}`;\n  }\n}\n\n/**\n * Factory function to create common rate limiters\n */\nexport class RateLimiterFactory {\n  /**\n   * GitHub API rate limiter (60 requests per hour for unauthenticated)\n   */\n  static createGitHubLimiter(): RateLimiter {\n    return new RateLimiter({\n      maxRequests: 60,\n      windowMs: 60 * 60 * 1000, // 1 hour\n      minDelayMs: 1000 // 1 second minimum between requests\n    });\n  }\n\n  /**\n   * Conservative rate limiter for update checks\n   * Allows 10 checks per hour with 30 second minimum delay\n   */\n  static createUpdateCheckLimiter(): RateLimiter {\n    return new RateLimiter({\n      maxRequests: 10,\n      windowMs: 60 * 60 * 1000, // 1 hour\n      minDelayMs: 30 * 1000 // 30 seconds between checks\n    });\n  }\n\n  /**\n   * Strict rate limiter for sensitive operations\n   * Allows 5 requests per hour with 1 minute minimum delay\n   */\n  static createStrictLimiter(): RateLimiter {\n    return new RateLimiter({\n      maxRequests: 5,\n      windowMs: 60 * 60 * 1000, // 1 hour\n      minDelayMs: 60 * 1000 // 1 minute between requests\n    });\n  }\n}"]}
|