@dollhousemcp/mcp-server 1.5.2 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/CHANGELOG.md +56 -0
  2. package/README.md +494 -111
  3. package/data/agents/code-reviewer.md +8 -1
  4. package/data/agents/research-assistant.md +8 -1
  5. package/data/agents/task-manager.md +8 -1
  6. package/data/ensembles/business-advisor.md +8 -1
  7. package/data/ensembles/creative-studio.md +8 -1
  8. package/data/ensembles/development-team.md +8 -1
  9. package/data/ensembles/security-analysis-team.md +8 -1
  10. package/data/memories/conversation-history.md +8 -1
  11. package/data/memories/learning-progress.md +8 -1
  12. package/data/memories/project-context.md +8 -1
  13. package/data/personas/business-consultant.md +8 -1
  14. package/data/personas/creative-writer.md +8 -1
  15. package/data/personas/debug-detective.md +8 -1
  16. package/data/personas/eli5-explainer.md +8 -1
  17. package/data/personas/security-analyst.md +8 -1
  18. package/data/personas/technical-analyst.md +8 -1
  19. package/data/skills/code-review.md +8 -1
  20. package/data/skills/creative-writing.md +8 -1
  21. package/data/skills/data-analysis.md +8 -1
  22. package/data/skills/penetration-testing.md +8 -1
  23. package/data/skills/research.md +8 -1
  24. package/data/skills/threat-modeling.md +8 -1
  25. package/data/skills/translation.md +8 -1
  26. package/data/templates/code-documentation.md +8 -1
  27. package/data/templates/email-professional.md +8 -1
  28. package/data/templates/meeting-notes.md +8 -1
  29. package/data/templates/penetration-test-report.md +8 -1
  30. package/data/templates/project-brief.md +8 -1
  31. package/data/templates/report-executive.md +8 -1
  32. package/data/templates/security-vulnerability-report.md +8 -1
  33. package/data/templates/threat-assessment-report.md +8 -1
  34. package/dist/auth/GitHubAuthManager.d.ts +6 -1
  35. package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
  36. package/dist/auth/GitHubAuthManager.js +45 -18
  37. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
  38. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
  39. package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
  40. package/dist/cache/CollectionCache.d.ts.map +1 -1
  41. package/dist/cache/CollectionCache.js +13 -3
  42. package/dist/cache/CollectionIndexCache.d.ts +77 -0
  43. package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
  44. package/dist/cache/CollectionIndexCache.js +349 -0
  45. package/dist/cache/LRUCache.d.ts +93 -0
  46. package/dist/cache/LRUCache.d.ts.map +1 -0
  47. package/dist/cache/LRUCache.js +299 -0
  48. package/dist/cache/index.d.ts +1 -0
  49. package/dist/cache/index.d.ts.map +1 -1
  50. package/dist/cache/index.js +2 -1
  51. package/dist/collection/CollectionBrowser.d.ts +21 -1
  52. package/dist/collection/CollectionBrowser.d.ts.map +1 -1
  53. package/dist/collection/CollectionBrowser.js +130 -10
  54. package/dist/collection/CollectionIndexManager.d.ts +151 -0
  55. package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
  56. package/dist/collection/CollectionIndexManager.js +499 -0
  57. package/dist/collection/CollectionSearch.d.ts +55 -0
  58. package/dist/collection/CollectionSearch.d.ts.map +1 -1
  59. package/dist/collection/CollectionSearch.js +338 -13
  60. package/dist/collection/CollectionSeeder.d.ts.map +1 -1
  61. package/dist/collection/CollectionSeeder.js +38 -1
  62. package/dist/collection/ElementInstaller.d.ts +31 -0
  63. package/dist/collection/ElementInstaller.d.ts.map +1 -1
  64. package/dist/collection/ElementInstaller.js +77 -15
  65. package/dist/collection/PersonaSubmitter.d.ts +1 -1
  66. package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
  67. package/dist/collection/PersonaSubmitter.js +2 -2
  68. package/dist/collection/index.d.ts +1 -0
  69. package/dist/collection/index.d.ts.map +1 -1
  70. package/dist/collection/index.js +2 -1
  71. package/dist/config/ConfigManager.d.ts +78 -0
  72. package/dist/config/ConfigManager.d.ts.map +1 -0
  73. package/dist/config/ConfigManager.js +216 -0
  74. package/dist/config/element-types.d.ts +135 -0
  75. package/dist/config/element-types.d.ts.map +1 -0
  76. package/dist/config/element-types.js +108 -0
  77. package/dist/config/index.d.ts +2 -0
  78. package/dist/config/index.d.ts.map +1 -1
  79. package/dist/config/index.js +3 -1
  80. package/dist/config/portfolio-constants.d.ts +83 -0
  81. package/dist/config/portfolio-constants.d.ts.map +1 -0
  82. package/dist/config/portfolio-constants.js +99 -0
  83. package/dist/elements/BaseElement.d.ts +14 -2
  84. package/dist/elements/BaseElement.d.ts.map +1 -1
  85. package/dist/elements/BaseElement.js +88 -6
  86. package/dist/elements/agents/Agent.d.ts +10 -1
  87. package/dist/elements/agents/Agent.d.ts.map +1 -1
  88. package/dist/elements/agents/Agent.js +66 -19
  89. package/dist/elements/agents/AgentManager.d.ts +2 -0
  90. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  91. package/dist/elements/agents/AgentManager.js +12 -10
  92. package/dist/elements/skills/Skill.d.ts +10 -1
  93. package/dist/elements/skills/Skill.d.ts.map +1 -1
  94. package/dist/elements/skills/Skill.js +40 -3
  95. package/dist/elements/skills/SkillManager.d.ts +1 -0
  96. package/dist/elements/skills/SkillManager.d.ts.map +1 -1
  97. package/dist/elements/skills/SkillManager.js +10 -4
  98. package/dist/elements/templates/Template.d.ts +10 -1
  99. package/dist/elements/templates/Template.d.ts.map +1 -1
  100. package/dist/elements/templates/Template.js +35 -18
  101. package/dist/elements/templates/TemplateManager.d.ts +1 -1
  102. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  103. package/dist/elements/templates/TemplateManager.js +6 -5
  104. package/dist/generated/version.d.ts +2 -2
  105. package/dist/generated/version.js +3 -3
  106. package/dist/index.barrel.d.ts +1 -2
  107. package/dist/index.barrel.d.ts.map +1 -1
  108. package/dist/index.barrel.js +2 -4
  109. package/dist/index.d.ts +143 -25
  110. package/dist/index.d.ts.map +1 -1
  111. package/dist/index.js +1883 -310
  112. package/dist/persona/PersonaElement.d.ts +10 -0
  113. package/dist/persona/PersonaElement.d.ts.map +1 -1
  114. package/dist/persona/PersonaElement.js +55 -32
  115. package/dist/persona/PersonaElementManager.d.ts.map +1 -1
  116. package/dist/persona/PersonaElementManager.js +13 -11
  117. package/dist/persona/PersonaLoader.d.ts.map +1 -1
  118. package/dist/persona/PersonaLoader.js +8 -2
  119. package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
  120. package/dist/persona/export-import/PersonaImporter.js +24 -5
  121. package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
  122. package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
  123. package/dist/persona/export-import/PersonaSharer.js +198 -22
  124. package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
  125. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  126. package/dist/portfolio/DefaultElementProvider.js +499 -7
  127. package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
  128. package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
  129. package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
  130. package/dist/portfolio/MigrationManager.d.ts.map +1 -1
  131. package/dist/portfolio/MigrationManager.js +136 -3
  132. package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
  133. package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
  134. package/dist/portfolio/PortfolioIndexManager.js +478 -0
  135. package/dist/portfolio/PortfolioManager.d.ts +5 -0
  136. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  137. package/dist/portfolio/PortfolioManager.js +61 -20
  138. package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
  139. package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
  140. package/dist/portfolio/PortfolioRepoManager.js +337 -0
  141. package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
  142. package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
  143. package/dist/portfolio/UnifiedIndexManager.js +1434 -0
  144. package/dist/portfolio/index.d.ts +15 -0
  145. package/dist/portfolio/index.d.ts.map +1 -0
  146. package/dist/portfolio/index.js +15 -0
  147. package/dist/portfolio/types.d.ts +7 -0
  148. package/dist/portfolio/types.d.ts.map +1 -1
  149. package/dist/portfolio/types.js +6 -1
  150. package/dist/security/InputValidator.d.ts.map +1 -1
  151. package/dist/security/InputValidator.js +50 -48
  152. package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
  153. package/dist/security/audit/SecurityAuditor.js +17 -9
  154. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  155. package/dist/security/audit/config/suppressions.js +19 -3
  156. package/dist/security/contentValidator.d.ts +2 -0
  157. package/dist/security/contentValidator.d.ts.map +1 -1
  158. package/dist/security/contentValidator.js +115 -4
  159. package/dist/security/secureYamlParser.d.ts +1 -0
  160. package/dist/security/secureYamlParser.d.ts.map +1 -1
  161. package/dist/security/secureYamlParser.js +29 -7
  162. package/dist/security/securityMonitor.d.ts +1 -1
  163. package/dist/security/securityMonitor.d.ts.map +1 -1
  164. package/dist/security/securityMonitor.js +1 -1
  165. package/dist/security/tokenManager.d.ts +1 -1
  166. package/dist/security/tokenManager.d.ts.map +1 -1
  167. package/dist/security/tokenManager.js +30 -10
  168. package/dist/server/ServerSetup.d.ts +22 -2
  169. package/dist/server/ServerSetup.d.ts.map +1 -1
  170. package/dist/server/ServerSetup.js +77 -12
  171. package/dist/server/tools/AuthTools.d.ts.map +1 -1
  172. package/dist/server/tools/AuthTools.js +33 -1
  173. package/dist/server/tools/BuildInfoTools.d.ts +25 -0
  174. package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
  175. package/dist/server/tools/BuildInfoTools.js +36 -0
  176. package/dist/server/tools/CollectionTools.d.ts.map +1 -1
  177. package/dist/server/tools/CollectionTools.js +55 -46
  178. package/dist/server/tools/ConfigTools.d.ts.map +1 -1
  179. package/dist/server/tools/ConfigTools.js +29 -1
  180. package/dist/server/tools/PersonaTools.d.ts +4 -2
  181. package/dist/server/tools/PersonaTools.d.ts.map +1 -1
  182. package/dist/server/tools/PersonaTools.js +5 -152
  183. package/dist/server/tools/PortfolioTools.d.ts +12 -0
  184. package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
  185. package/dist/server/tools/PortfolioTools.js +221 -0
  186. package/dist/server/tools/index.d.ts +3 -1
  187. package/dist/server/tools/index.d.ts.map +1 -1
  188. package/dist/server/tools/index.js +4 -2
  189. package/dist/server/types.d.ts +40 -5
  190. package/dist/server/types.d.ts.map +1 -1
  191. package/dist/server/types.js +1 -1
  192. package/dist/services/BuildInfoService.d.ts +84 -0
  193. package/dist/services/BuildInfoService.d.ts.map +1 -0
  194. package/dist/services/BuildInfoService.js +271 -0
  195. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
  196. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
  197. package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
  198. package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
  199. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
  200. package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
  201. package/dist/tools/portfolio/types.d.ts +41 -0
  202. package/dist/tools/portfolio/types.d.ts.map +1 -0
  203. package/dist/tools/portfolio/types.js +15 -0
  204. package/dist/types/collection.d.ts +51 -0
  205. package/dist/types/collection.d.ts.map +1 -1
  206. package/dist/types/collection.js +1 -1
  207. package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
  208. package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
  209. package/dist/utils/EarlyTerminationSearch.js +164 -0
  210. package/dist/utils/ErrorHandler.d.ts +86 -0
  211. package/dist/utils/ErrorHandler.d.ts.map +1 -0
  212. package/dist/utils/ErrorHandler.js +201 -0
  213. package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
  214. package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
  215. package/dist/utils/FileDiscoveryUtil.js +169 -0
  216. package/dist/utils/GitHubRateLimiter.d.ts +88 -0
  217. package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
  218. package/dist/utils/GitHubRateLimiter.js +315 -0
  219. package/dist/utils/PerformanceMonitor.d.ts +134 -0
  220. package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
  221. package/dist/utils/PerformanceMonitor.js +347 -0
  222. package/dist/utils/RateLimiter.d.ts.map +1 -0
  223. package/dist/utils/RateLimiter.js +172 -0
  224. package/dist/utils/SecureDownloader.d.ts +241 -0
  225. package/dist/utils/SecureDownloader.d.ts.map +1 -0
  226. package/dist/utils/SecureDownloader.js +759 -0
  227. package/dist/utils/ToolCache.d.ts +82 -0
  228. package/dist/utils/ToolCache.d.ts.map +1 -0
  229. package/dist/utils/ToolCache.js +196 -0
  230. package/dist/utils/errorCodes.d.ts +136 -0
  231. package/dist/utils/errorCodes.d.ts.map +1 -0
  232. package/dist/utils/errorCodes.js +87 -0
  233. package/dist/utils/index.d.ts +3 -0
  234. package/dist/utils/index.d.ts.map +1 -1
  235. package/dist/utils/index.js +4 -1
  236. package/dist/utils/installation.d.ts +1 -1
  237. package/dist/utils/installation.d.ts.map +1 -1
  238. package/dist/utils/installation.js +9 -8
  239. package/dist/utils/searchUtils.d.ts +31 -0
  240. package/dist/utils/searchUtils.d.ts.map +1 -1
  241. package/dist/utils/searchUtils.js +62 -1
  242. package/package.json +17 -7
  243. package/dist/config/updateConfig.d.ts +0 -84
  244. package/dist/config/updateConfig.d.ts.map +0 -1
  245. package/dist/config/updateConfig.js +0 -148
  246. package/dist/server/tools/UpdateTools.d.ts +0 -10
  247. package/dist/server/tools/UpdateTools.d.ts.map +0 -1
  248. package/dist/server/tools/UpdateTools.js +0 -85
  249. package/dist/update/BackupManager.d.ts +0 -63
  250. package/dist/update/BackupManager.d.ts.map +0 -1
  251. package/dist/update/BackupManager.js +0 -370
  252. package/dist/update/DependencyChecker.d.ts +0 -41
  253. package/dist/update/DependencyChecker.d.ts.map +0 -1
  254. package/dist/update/DependencyChecker.js +0 -132
  255. package/dist/update/RateLimiter.d.ts.map +0 -1
  256. package/dist/update/RateLimiter.js +0 -172
  257. package/dist/update/SignatureVerifier.d.ts +0 -71
  258. package/dist/update/SignatureVerifier.d.ts.map +0 -1
  259. package/dist/update/SignatureVerifier.js +0 -214
  260. package/dist/update/UpdateChecker.d.ts +0 -132
  261. package/dist/update/UpdateChecker.d.ts.map +0 -1
  262. package/dist/update/UpdateChecker.js +0 -506
  263. package/dist/update/UpdateManager.d.ts +0 -60
  264. package/dist/update/UpdateManager.d.ts.map +0 -1
  265. package/dist/update/UpdateManager.js +0 -730
  266. package/dist/update/VersionManager.d.ts +0 -31
  267. package/dist/update/VersionManager.d.ts.map +0 -1
  268. package/dist/update/VersionManager.js +0 -181
  269. package/dist/update/index.d.ts +0 -9
  270. package/dist/update/index.d.ts.map +0 -1
  271. package/dist/update/index.js +0 -9
  272. /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
@@ -0,0 +1,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":"AAAA;;;;;;;;GAQG;AAeH,MAAM,OAAO,WAAW;IACd,MAAM,CAAS;IACf,UAAU,CAAS;IACnB,WAAW,CAAS;IACX,SAAS,CAAS;IAClB,UAAU,CAAS;IACnB,QAAQ,CAAS;IAElC,YAAY,MAAyB;QACnC,IAAI,MAAM,CAAC,WAAW,IAAI,CAAC,EAAE,CAAC;YAC5B,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;QAClD,CAAC;QACD,IAAI,MAAM,CAAC,QAAQ,IAAI,CAAC,EAAE,CAAC;YACzB,MAAM,IAAI,KAAK,CAAC,2BAA2B,CAAC,CAAC;QAC/C,CAAC;QAED,IAAI,CAAC,SAAS,GAAG,MAAM,CAAC,WAAW,CAAC;QACpC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC;QAC7B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,SAAS,GAAG,MAAM,CAAC,QAAQ,CAAC;QAEnD,mDAAmD;QACnD,IAAI,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACvD,MAAM,IAAI,KAAK,CAAC,gEAAgE,CAAC,CAAC;QACpF,CAAC;QAED,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC7B,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;QACrB,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC,UAAU,IAAI,CAAC,CAAC;IACzC,CAAC;IAED;;;OAGG;IACH,UAAU;QACR,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAEvB,sCAAsC;QACtC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;QAEvB,uCAAuC;QACvC,IAAI,IAAI,CAAC,QAAQ,GAAG,CAAC,IAAI,IAAI,CAAC,WAAW,GAAG,CAAC,EAAE,CAAC;YAC9C,MAAM,oBAAoB,GAAG,GAAG,GAAG,IAAI,CAAC,WAAW,CAAC;YACpD,IAAI,oBAAoB,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACzC,MAAM,YAAY,GAAG,IAAI,CAAC,QAAQ,GAAG,oBAAoB,CAAC;gBAC1D,OAAO;oBACL,OAAO,EAAE,KAAK;oBACd,YAAY;oBACZ,eAAe,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC;oBACxC,SAAS,EAAE,IAAI,IAAI,CAAC,GAAG,GAAG,YAAY,CAAC;iBACxC,CAAC;YACJ,CAAC;QACH,CAAC;QAED,oCAAoC;QACpC,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpB,kDAAkD;YAClD,MAAM,YAAY,GAAG,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC;YACrC,MAAM,gBAAgB,GAAG,YAAY,GAAG,IAAI,CAAC,UAAU,CAAC;YAExD,OAAO;gBACL,OAAO,EAAE,KAAK;gBACd,YAAY,EAAE,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC;gBACzC,eAAe,EAAE,CAAC;gBAClB,SAAS,EAAE,IAAI,IAAI,CAAC,GAAG,GAAG,gBAAgB,CAAC;aAC5C,CAAC;QACJ,CAAC;QAED,qBAAqB;QACrB,OAAO;YACL,OAAO,EAAE,IAAI;YACb,eAAe,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC;YACxC,SAAS,EAAE,IAAI,CAAC,YAAY,EAAE;SAC/B,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,YAAY;QACV,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;QAEvB,IAAI,IAAI,CAAC,MAAM,IAAI,CAAC,EAAE,CAAC;YACrB,IAAI,CAAC,MAAM,IAAI,CAAC,CAAC;YACjB,IAAI,CAAC,WAAW,GAAG,GAAG,CAAC;QACzB,CAAC;IACH,CAAC;IAED;;OAEG;IACH,SAAS;QACP,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;QAEvB,OAAO;YACL,OAAO,EAAE,IAAI,CAAC,MAAM,IAAI,CAAC;YACzB,eAAe,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC;YACxC,SAAS,EAAE,IAAI,CAAC,YAAY,EAAE;SAC/B,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,KAAK;QACH,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC;QAC7B,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC7B,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;IACvB,CAAC;IAED;;OAEG;IACK,YAAY,CAAC,GAAW;QAC9B,MAAM,mBAAmB,GAAG,GAAG,GAAG,IAAI,CAAC,UAAU,CAAC;QAClD,MAAM,WAAW,GAAG,mBAAmB,GAAG,IAAI,CAAC,UAAU,CAAC;QAE1D,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,MAAM,GAAG,WAAW,CAAC,CAAC;QAClE,IAAI,CAAC,UAAU,GAAG,GAAG,CAAC;IACxB,CAAC;IAED;;OAEG;IACK,YAAY;QAClB,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,MAAM,YAAY,GAAG,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,MAAM,CAAC;QAClD,MAAM,WAAW,GAAG,YAAY,GAAG,IAAI,CAAC,UAAU,CAAC;QACnD,OAAO,IAAI,IAAI,CAAC,GAAG,GAAG,WAAW,CAAC,CAAC;IACrC,CAAC;IAED;;OAEG;IACH,QAAQ;QACN,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;QAChC,OAAO,cAAc,MAAM,CAAC,eAAe,IAAI,IAAI,CAAC,SAAS,WAAW;YACjE,aAAa,MAAM,CAAC,SAAS,CAAC,WAAW,EAAE,EAAE,CAAC;IACvD,CAAC;CACF;AAED;;GAEG;AACH,MAAM,OAAO,kBAAkB;IAC7B;;OAEG;IACH,MAAM,CAAC,mBAAmB;QACxB,OAAO,IAAI,WAAW,CAAC;YACrB,WAAW,EAAE,EAAE;YACf,QAAQ,EAAE,EAAE,GAAG,EAAE,GAAG,IAAI,EAAE,SAAS;YACnC,UAAU,EAAE,IAAI,CAAC,oCAAoC;SACtD,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH,MAAM,CAAC,wBAAwB;QAC7B,OAAO,IAAI,WAAW,CAAC;YACrB,WAAW,EAAE,EAAE;YACf,QAAQ,EAAE,EAAE,GAAG,EAAE,GAAG,IAAI,EAAE,SAAS;YACnC,UAAU,EAAE,EAAE,GAAG,IAAI,CAAC,4BAA4B;SACnD,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH,MAAM,CAAC,mBAAmB;QACxB,OAAO,IAAI,WAAW,CAAC;YACrB,WAAW,EAAE,CAAC;YACd,QAAQ,EAAE,EAAE,GAAG,EAAE,GAAG,IAAI,EAAE,SAAS;YACnC,UAAU,EAAE,EAAE,GAAG,IAAI,CAAC,4BAA4B;SACnD,CAAC,CAAC;IACL,CAAC;CACF","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}"]}