@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,315 @@
1
+ /**
2
+ * GitHubRateLimiter - Specialized rate limiter for GitHub API calls
3
+ *
4
+ * Features:
5
+ * - Respects GitHub's authenticated (5000/hour) and unauthenticated (60/hour) limits
6
+ * - Client-side queuing when approaching limits
7
+ * - Request prioritization for critical operations
8
+ * - Comprehensive logging for quota management
9
+ * - Early termination when exact matches are found
10
+ */
11
+ import { RateLimiter } from './RateLimiter.js';
12
+ import { GITHUB_API_RATE_LIMITS } from '../config/portfolio-constants.js';
13
+ import { TokenManager } from '../security/tokenManager.js';
14
+ import { logger } from './logger.js';
15
+ import { SecurityMonitor } from '../security/securityMonitor.js';
16
+ import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
17
+ export class GitHubRateLimiter {
18
+ rateLimiter;
19
+ requestQueue = [];
20
+ processing = false;
21
+ lastRateLimitInfo;
22
+ isAuthenticated = false;
23
+ constructor() {
24
+ // Initialize with conservative limits - will update based on auth status
25
+ this.updateLimitsForAuthStatus();
26
+ this.setupPeriodicStatusCheck();
27
+ }
28
+ /**
29
+ * Update rate limits based on current authentication status
30
+ */
31
+ async updateLimitsForAuthStatus() {
32
+ try {
33
+ const token = await TokenManager.getGitHubTokenAsync();
34
+ const newIsAuthenticated = !!token;
35
+ // Only recreate rate limiter if auth status changed
36
+ if (newIsAuthenticated !== this.isAuthenticated) {
37
+ this.isAuthenticated = newIsAuthenticated;
38
+ const limit = this.isAuthenticated
39
+ ? GITHUB_API_RATE_LIMITS.AUTHENTICATED_LIMIT
40
+ : GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT;
41
+ // Apply buffer to stay below actual limits
42
+ const bufferedLimit = Math.floor(limit * GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE);
43
+ const config = {
44
+ maxRequests: bufferedLimit,
45
+ windowMs: GITHUB_API_RATE_LIMITS.WINDOW_MS,
46
+ minDelayMs: GITHUB_API_RATE_LIMITS.MIN_DELAY_MS
47
+ };
48
+ this.rateLimiter = new RateLimiter(config);
49
+ logger.info('GitHub rate limiter updated', {
50
+ authenticated: this.isAuthenticated,
51
+ limit: bufferedLimit,
52
+ originalLimit: limit,
53
+ bufferPercentage: GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE
54
+ });
55
+ }
56
+ }
57
+ catch (error) {
58
+ logger.warn('Failed to check authentication status for rate limiting', { error });
59
+ // Fall back to unauthenticated limits
60
+ this.isAuthenticated = false;
61
+ this.rateLimiter = new RateLimiter({
62
+ maxRequests: Math.floor(GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT * GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE),
63
+ windowMs: GITHUB_API_RATE_LIMITS.WINDOW_MS,
64
+ minDelayMs: GITHUB_API_RATE_LIMITS.MIN_DELAY_MS
65
+ });
66
+ }
67
+ }
68
+ /**
69
+ * Setup periodic check for rate limit status
70
+ */
71
+ setupPeriodicStatusCheck() {
72
+ // Check auth status every 5 minutes
73
+ setInterval(() => {
74
+ this.updateLimitsForAuthStatus().catch(error => {
75
+ logger.warn('Periodic auth status check failed', { error });
76
+ });
77
+ }, 5 * 60 * 1000);
78
+ }
79
+ /**
80
+ * Queue a GitHub API request with rate limiting
81
+ * @param operation Description of the operation
82
+ * @param apiCall Function that makes the actual API call
83
+ * @param priority Request priority (high, normal, low)
84
+ * @returns Promise that resolves with the API response
85
+ */
86
+ async queueRequest(operation, apiCall, priority = 'normal') {
87
+ // SECURITY FIX (DMCP-SEC-004): Normalize Unicode in operation name to prevent injection attacks
88
+ const normalizedOperation = UnicodeValidator.normalize(operation);
89
+ if (!normalizedOperation.isValid) {
90
+ SecurityMonitor.logSecurityEvent({
91
+ type: 'UNICODE_VALIDATION_ERROR',
92
+ severity: 'MEDIUM',
93
+ source: 'GitHubRateLimiter.queueRequest',
94
+ details: `Invalid Unicode in operation name: ${normalizedOperation.detectedIssues?.[0] || 'unknown error'}`
95
+ });
96
+ // Use a safe fallback for the operation name
97
+ operation = 'github-api-request';
98
+ }
99
+ else {
100
+ operation = normalizedOperation.normalizedContent;
101
+ }
102
+ const requestId = `${operation}-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;
103
+ return new Promise((resolve, reject) => {
104
+ const request = {
105
+ id: requestId,
106
+ operation,
107
+ priority,
108
+ timestamp: Date.now(),
109
+ resolve: async (value) => {
110
+ try {
111
+ logger.debug('Executing GitHub API request', {
112
+ operation,
113
+ requestId,
114
+ queueWaitTime: Date.now() - request.timestamp
115
+ });
116
+ const result = await apiCall();
117
+ resolve(result);
118
+ // Log successful API usage for quota tracking
119
+ this.logApiUsage(operation, 'success');
120
+ }
121
+ catch (error) {
122
+ // Check if this is a rate limit error from GitHub
123
+ if (this.isGitHubRateLimitError(error)) {
124
+ this.handleGitHubRateLimit(error);
125
+ }
126
+ reject(error);
127
+ this.logApiUsage(operation, 'error', error);
128
+ }
129
+ },
130
+ reject
131
+ };
132
+ // Add to queue with priority ordering
133
+ this.addToQueue(request);
134
+ this.processQueue();
135
+ });
136
+ }
137
+ /**
138
+ * Add request to queue with priority ordering
139
+ */
140
+ addToQueue(request) {
141
+ // Insert based on priority: high > normal > low
142
+ // Within same priority, maintain FIFO order
143
+ const priorityOrder = { high: 0, normal: 1, low: 2 };
144
+ let insertIndex = this.requestQueue.length;
145
+ for (let i = 0; i < this.requestQueue.length; i++) {
146
+ if (priorityOrder[request.priority] < priorityOrder[this.requestQueue[i].priority]) {
147
+ insertIndex = i;
148
+ break;
149
+ }
150
+ }
151
+ this.requestQueue.splice(insertIndex, 0, request);
152
+ logger.debug('GitHub API request queued', {
153
+ operation: request.operation,
154
+ priority: request.priority,
155
+ queuePosition: insertIndex,
156
+ totalQueued: this.requestQueue.length
157
+ });
158
+ }
159
+ /**
160
+ * Process the request queue
161
+ */
162
+ async processQueue() {
163
+ if (this.processing || this.requestQueue.length === 0) {
164
+ return;
165
+ }
166
+ this.processing = true;
167
+ try {
168
+ while (this.requestQueue.length > 0) {
169
+ // Update auth status periodically
170
+ if (Math.random() < 0.1) { // 10% chance
171
+ await this.updateLimitsForAuthStatus();
172
+ }
173
+ const rateLimitStatus = this.rateLimiter.checkLimit();
174
+ if (!rateLimitStatus.allowed) {
175
+ // Log rate limit wait
176
+ logger.info('GitHub API rate limit reached, waiting', {
177
+ retryAfterMs: rateLimitStatus.retryAfterMs,
178
+ remainingTokens: rateLimitStatus.remainingTokens,
179
+ queueLength: this.requestQueue.length,
180
+ resetTime: rateLimitStatus.resetTime
181
+ });
182
+ // Wait for the specified time
183
+ await new Promise(resolve => setTimeout(resolve, rateLimitStatus.retryAfterMs || 1000));
184
+ continue;
185
+ }
186
+ // Process the next request
187
+ const request = this.requestQueue.shift();
188
+ this.rateLimiter.consumeToken();
189
+ // Execute the request
190
+ request.resolve(null); // This will trigger the actual API call
191
+ }
192
+ }
193
+ finally {
194
+ this.processing = false;
195
+ }
196
+ }
197
+ /**
198
+ * Get current rate limit status
199
+ */
200
+ getStatus() {
201
+ const baseStatus = this.rateLimiter.getStatus();
202
+ return {
203
+ ...baseStatus,
204
+ queueLength: this.requestQueue.length,
205
+ currentLimit: this.isAuthenticated
206
+ ? GITHUB_API_RATE_LIMITS.AUTHENTICATED_LIMIT
207
+ : GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT,
208
+ rateLimitInfo: this.lastRateLimitInfo
209
+ };
210
+ }
211
+ /**
212
+ * Check if an error is a GitHub rate limit error
213
+ */
214
+ isGitHubRateLimitError(error) {
215
+ return error?.status === 429 ||
216
+ error?.response?.status === 429 ||
217
+ (typeof error?.message === 'string' && error.message.toLowerCase().includes('rate limit'));
218
+ }
219
+ /**
220
+ * Handle GitHub rate limit error response
221
+ */
222
+ handleGitHubRateLimit(error) {
223
+ let resetTime;
224
+ let remainingRequests = 0;
225
+ // Parse rate limit headers if available
226
+ if (error?.response?.headers) {
227
+ const headers = error.response.headers;
228
+ const resetTimestamp = parseInt(headers['x-ratelimit-reset'] || '0');
229
+ const remaining = parseInt(headers['x-ratelimit-remaining'] || '0');
230
+ const limit = parseInt(headers['x-ratelimit-limit'] || '0');
231
+ if (resetTimestamp > 0) {
232
+ resetTime = new Date(resetTimestamp * 1000);
233
+ }
234
+ this.lastRateLimitInfo = {
235
+ limit,
236
+ remaining,
237
+ reset: resetTime || new Date(Date.now() + 60 * 60 * 1000), // Default to 1 hour
238
+ used: limit - remaining
239
+ };
240
+ remainingRequests = remaining;
241
+ }
242
+ logger.warn('GitHub API rate limit hit from server', {
243
+ remaining: remainingRequests,
244
+ resetTime,
245
+ queueLength: this.requestQueue.length,
246
+ errorMessage: error?.message
247
+ });
248
+ // Log as a security event for monitoring
249
+ SecurityMonitor.logSecurityEvent({
250
+ type: 'RATE_LIMIT_EXCEEDED',
251
+ severity: 'MEDIUM',
252
+ source: 'GitHubRateLimiter.handleGitHubRateLimit',
253
+ details: `GitHub API rate limit exceeded. Remaining: ${remainingRequests}, Queue: ${this.requestQueue.length}`,
254
+ metadata: {
255
+ rateLimitInfo: this.lastRateLimitInfo,
256
+ authenticated: this.isAuthenticated
257
+ }
258
+ });
259
+ }
260
+ /**
261
+ * Log API usage for monitoring and diagnostics
262
+ */
263
+ logApiUsage(operation, result, error) {
264
+ const status = this.getStatus();
265
+ logger.debug('GitHub API usage logged', {
266
+ operation,
267
+ result,
268
+ remainingTokens: status.remainingTokens,
269
+ queueLength: status.queueLength,
270
+ authenticated: this.isAuthenticated,
271
+ error: error?.message
272
+ });
273
+ // Log warning if getting close to rate limits
274
+ if (status.remainingTokens < 100 && this.isAuthenticated) {
275
+ logger.warn('Approaching GitHub API rate limit', {
276
+ operation,
277
+ remainingTokens: status.remainingTokens,
278
+ currentLimit: status.currentLimit,
279
+ recommendation: 'Consider reducing API usage frequency'
280
+ });
281
+ }
282
+ else if (status.remainingTokens < 10 && !this.isAuthenticated) {
283
+ logger.warn('Approaching GitHub API rate limit (unauthenticated)', {
284
+ operation,
285
+ remainingTokens: status.remainingTokens,
286
+ currentLimit: status.currentLimit,
287
+ recommendation: 'Consider authenticating for higher rate limits'
288
+ });
289
+ }
290
+ }
291
+ /**
292
+ * Clear the request queue (for testing or emergency situations)
293
+ */
294
+ clearQueue() {
295
+ const clearedCount = this.requestQueue.length;
296
+ // Reject all pending requests
297
+ this.requestQueue.forEach(request => {
298
+ request.reject(new Error('Request queue cleared'));
299
+ });
300
+ this.requestQueue = [];
301
+ logger.info('GitHub API request queue cleared', { clearedCount });
302
+ }
303
+ /**
304
+ * Reset the rate limiter (for testing)
305
+ */
306
+ reset() {
307
+ this.rateLimiter.reset();
308
+ this.clearQueue();
309
+ this.processing = false;
310
+ logger.info('GitHub rate limiter reset');
311
+ }
312
+ }
313
+ // Singleton instance for global use
314
+ export const githubRateLimiter = new GitHubRateLimiter();
315
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiR2l0SHViUmF0ZUxpbWl0ZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdXRpbHMvR2l0SHViUmF0ZUxpbWl0ZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7OztHQVNHO0FBRUgsT0FBTyxFQUFFLFdBQVcsRUFBc0MsTUFBTSxrQkFBa0IsQ0FBQztBQUNuRixPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUMxRSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDM0QsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUNyQyxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sZ0NBQWdDLENBQUM7QUFDakUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sNENBQTRDLENBQUM7QUF3QjlFLE1BQU0sT0FBTyxpQkFBaUI7SUFDcEIsV0FBVyxDQUFlO0lBQzFCLFlBQVksR0FBdUIsRUFBRSxDQUFDO0lBQ3RDLFVBQVUsR0FBRyxLQUFLLENBQUM7SUFDbkIsaUJBQWlCLENBQXVCO0lBQ3hDLGVBQWUsR0FBRyxLQUFLLENBQUM7SUFFaEM7UUFDRSx5RUFBeUU7UUFDekUsSUFBSSxDQUFDLHlCQUF5QixFQUFFLENBQUM7UUFDakMsSUFBSSxDQUFDLHdCQUF3QixFQUFFLENBQUM7SUFDbEMsQ0FBQztJQUVEOztPQUVHO0lBQ0ssS0FBSyxDQUFDLHlCQUF5QjtRQUNyQyxJQUFJLENBQUM7WUFDSCxNQUFNLEtBQUssR0FBRyxNQUFNLFlBQVksQ0FBQyxtQkFBbUIsRUFBRSxDQUFDO1lBQ3ZELE1BQU0sa0JBQWtCLEdBQUcsQ0FBQyxDQUFDLEtBQUssQ0FBQztZQUVuQyxvREFBb0Q7WUFDcEQsSUFBSSxrQkFBa0IsS0FBSyxJQUFJLENBQUMsZUFBZSxFQUFFLENBQUM7Z0JBQ2hELElBQUksQ0FBQyxlQUFlLEdBQUcsa0JBQWtCLENBQUM7Z0JBRTFDLE1BQU0s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LE9BQU8sQ0FBQyxNQUFNLENBQUMsSUFBSSxLQUFLLENBQUMsdUJBQXVCLENBQUMsQ0FBQyxDQUFDO1FBQ3JELENBQUMsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDLFlBQVksR0FBRyxFQUFFLENBQUM7UUFFdkIsTUFBTSxDQUFDLElBQUksQ0FBQyxrQ0FBa0MsRUFBRSxFQUFFLFlBQVksRUFBRSxDQUFDLENBQUM7SUFDcEUsQ0FBQztJQUVEOztPQUVHO0lBQ0gsS0FBSztRQUNILElBQUksQ0FBQyxXQUFXLENBQUMsS0FBSyxFQUFFLENBQUM7UUFDekIsSUFBSSxDQUFDLFVBQVUsRUFBRSxDQUFDO1FBQ2xCLElBQUksQ0FBQyxVQUFVLEdBQUcsS0FBSyxDQUFDO1FBQ3hCLE1BQU0sQ0FBQyxJQUFJLENBQUMsMkJBQTJCLENBQUMsQ0FBQztJQUMzQyxDQUFDO0NBQ0Y7QUFFRCxvQ0FBb0M7QUFDcEMsTUFBTSxDQUFDLE1BQU0saUJBQWlCLEdBQUcsSUFBSSxpQkFBaUIsRUFBRSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHaXRIdWJSYXRlTGltaXRlciAtIFNwZWNpYWxpemVkIHJhdGUgbGltaXRlciBmb3IgR2l0SHViIEFQSSBjYWxsc1xuICogXG4gKiBGZWF0dXJlczpcbiAqIC0gUmVzcGVjdHMgR2l0SHViJ3MgYXV0aGVudGljYXRlZCAoNTAwMC9ob3VyKSBhbmQgdW5hdXRoZW50aWNhdGVkICg2MC9ob3VyKSBsaW1pdHNcbiAqIC0gQ2xpZW50LXNpZGUgcXVldWluZyB3aGVuIGFwcHJvYWNoaW5nIGxpbWl0c1xuICogLSBSZXF1ZXN0IHByaW9yaXRpemF0aW9uIGZvciBjcml0aWNhbCBvcGVyYXRpb25zXG4gKiAtIENvbXByZWhlbnNpdmUgbG9nZ2luZyBmb3IgcXVvdGEgbWFuYWdlbWVudFxuICogLSBFYXJseSB0ZXJtaW5hdGlvbiB3aGVuIGV4YWN0IG1hdGNoZXMgYXJlIGZvdW5kXG4gKi9cblxuaW1wb3J0IHsgUmF0ZUxpbWl0ZXIsIFJhdGVMaW1pdGVyQ29uZmlnLCBSYXRlTGltaXRTdGF0dXMgfSBmcm9tICcuL1JhdGVMaW1pdGVyLmpzJztcbmltcG9ydCB7IEdJVEhVQl9BUElfUkFURV9MSU1JVFMgfSBmcm9tICcuLi9jb25maWcvcG9ydGZvbGlvLWNvbnN0YW50cy5qcyc7XG5pbXBvcnQgeyBUb2tlbk1hbmFnZXIgfSBmcm9tICcuLi9zZWN1cml0eS90b2tlbk1hbmFnZXIuanMnO1xuaW1wb3J0IHsgbG9nZ2VyIH0gZnJvbSAnLi9sb2dnZXIuanMnO1xuaW1wb3J0IHsgU2VjdXJpdHlNb25pdG9yIH0gZnJvbSAnLi4vc2VjdXJpdHkvc2VjdXJpdHlNb25pdG9yLmpzJztcbmltcG9ydCB7IFVuaWNvZGVWYWxpZGF0b3IgfSBmcm9tICcuLi9zZWN1cml0eS92YWxpZGF0b3JzL3VuaWNvZGVWYWxpZGF0b3IuanMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEdpdEh1YlJhdGVMaW1pdEluZm8ge1xuICBsaW1pdDogbnVtYmVyO1xuICByZW1haW5pbmc6IG51bWJlcjtcbiAgcmVzZXQ6IERhdGU7XG4gIHVzZWQ6IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBHaXRIdWJBcGlSZXF1ZXN0IHtcbiAgaWQ6IHN0cmluZztcbiAgb3BlcmF0aW9uOiBzdHJpbmc7XG4gIHByaW9yaXR5OiAnaGlnaCcgfCAnbm9ybWFsJyB8ICdsb3cnO1xuICB0aW1lc3RhbXA6IG51bWJlcjtcbiAgcmVzb2x2ZTogKHZhbHVlOiBhbnkpID0+IHZvaWQ7XG4gIHJlamVjdDogKGVycm9yOiBhbnkpID0+IHZvaWQ7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgR2l0SHViUmF0ZVN0YXR1cyBleHRlbmRzIFJhdGVMaW1pdFN0YXR1cyB7XG4gIHF1ZXVlTGVuZ3RoOiBudW1iZXI7XG4gIGN1cnJlbnRMaW1pdDogbnVtYmVyO1xuICByYXRlTGltaXRJbmZvPzogR2l0SHViUmF0ZUxpbWl0SW5mbztcbn1cblxuZXhwb3J0IGNsYXNzIEdpdEh1YlJhdGVMaW1pdGVyIHtcbiAgcHJpdmF0ZSByYXRlTGltaXRlciE6IFJhdGVMaW1pdGVyO1xuICBwcml2YXRlIHJlcXVlc3RRdWV1ZTogR2l0SHViQXBpUmVxdWVzdFtdID0gW107XG4gIHByaXZhdGUgcHJvY2Vzc2luZyA9IGZhbHNlO1xuICBwcml2YXRlIGxhc3RSYXRlTGltaXRJbmZvPzogR2l0SHViUmF0ZUxpbWl0SW5mbztcbiAgcHJpdmF0ZSBpc0F1dGhlbnRpY2F0ZWQgPSBmYWxzZTtcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgICAvLyBJbml0aWFsaXplIHdpdGggY29uc2VydmF0aXZlIGxpbWl0cyAtIHdpbGwgdXBkYXRlIGJhc2VkIG9uIGF1dGggc3RhdHVzXG4gICAgdGhpcy51cGRhdGVMaW1pdHNGb3JBdXRoU3RhdHVzKCk7XG4gICAgdGhpcy5zZXR1cFBlcmlvZGljU3RhdHVzQ2hlY2soKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBVcGRhdGUgcmF0ZSBsaW1pdHMgYmFzZWQgb24gY3VycmVudCBhdXRoZW50aWNhdGlvbiBzdGF0dXNcbiAgICovXG4gIHByaXZhdGUgYXN5bmMgdXBkYXRlTGltaXRzRm9yQXV0aFN0YXR1cygpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICB0cnkge1xuICAgICAgY29uc3QgdG9rZW4gPSBhd2FpdCBUb2tlbk1hbmFnZXIuZ2V0R2l0SHViVG9rZW5Bc3luYygpO1xuICAgICAgY29uc3QgbmV3SXNBdXRoZW50aWNhdGVkID0gISF0b2tlbjtcbiAgICAgIFxuICAgICAgLy8gT25seSByZWNyZWF0ZSByYXRlIGxpbWl0ZXIgaWYgYXV0aCBzdGF0dXMgY2hhbmdlZFxuICAgICAgaWYgKG5ld0lzQXV0aGVudGljYXRlZCAhPT0gdGhpcy5pc0F1dGhlbnRpY2F0ZWQpIHtcbiAgICAgICAgdGhpcy5pc0F1dGhlbnRpY2F0ZWQgPSBuZXdJc0F1dGhlbnRpY2F0ZWQ7XG4gICAgICAgIFxuICAgICAgICBjb25zdCBsaW1pdCA9IHRoaXMuaXNBdXRoZW50aWNhdGVkIFxuICAgICAgICAgID8gR0lUSFVCX0FQSV9SQVRFX0xJTUlUUy5BVVRIRU5USUNBVEVEX0xJTUlUIFxuICAgICAgICAgIDogR0lUSFVCX0FQSV9SQVRFX0xJTUlUUy5VTkFVVEhFTlRJQ0FURURfTElNSVQ7XG4gICAgICAgICAgXG4gICAgICAgIC8vIEFwcGx5IGJ1ZmZlciB0byBzdGF5IGJlbG93IGFjdHVhbCBsaW1pdHNcbiAgICAgICAgY29uc3QgYnVmZmVyZWRMaW1pdCA9IE1hdGguZmxvb3IobGltaXQgKiBHSVRIVUJfQVBJX1JBVEVfTElNSVRTLkJVRkZFUl9QRVJDRU5UQUdFKTtcbiAgICAgICAgXG4gICAgICAgIGNvbnN0IGNvbmZpZzogUmF0ZUxpbWl0ZXJDb25maWcgPSB7XG4gICAgICAgICAgbWF4UmVxdWVzdHM6IGJ1ZmZlcmVkTGltaXQsXG4gICAgICAgICAgd2luZG93TXM6IEdJVEhVQl9BUElfUkFURV9MSU1JVFMuV0lORE9XX01TLFxuICAgICAgICAgIG1pbkRlbGF5TXM6IEdJVEhVQl9BUElfUkFURV9MSU1JVFMuTUlOX0RFTEFZX01TXG4gICAgICAgIH07XG4gICAgICAgIFxuICAgICAgICB0aGlzLnJhdGVMaW1pdGVyID0gbmV3IFJhdGVMaW1pdGVyKGNvbmZpZyk7XG4gICAgICAgIFxuICAgICAgICBsb2dnZXIuaW5mbygnR2l0SHViIHJhdGUgbGltaXRlciB1cGRhdGVkJywge1xuICAgICAgICAgIGF1dGhlbnRpY2F0ZWQ6IHRoaXMuaXNBdXRoZW50aWNhdGVkLFxuICAgICAgICAgIGxpbWl0OiBidWZmZXJlZExpbWl0LFxuICAgICAgICAgIG9yaWdpbmFsTGltaXQ6IGxpbWl0LFxuICAgICAgICAgIGJ1ZmZlclBlcmNlbnRhZ2U6IEdJVEhVQl9BUElfUkFURV9MSU1JVFMuQlVGRkVSX1BFUkNFTlRBR0VcbiAgICAgICAgfSk7XG4gICAgICB9XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGxvZ2dlci53YXJuKCdGYWlsZWQgdG8gY2hlY2sgYXV0aGVudGljYXRpb24gc3RhdHVzIGZvciByYXRlIGxpbWl0aW5nJywgeyBlcnJvciB9KTtcbiAgICAgIC8vIEZhbGwgYmFjayB0byB1bmF1dGhlbnRpY2F0ZWQgbGltaXRzXG4gICAgICB0aGlzLmlzQXV0aGVudGljYXRlZCA9IGZhbHNlO1xuICAgICAgdGhpcy5yYXRlTGltaXRlciA9IG5ldyBSYXRlTGltaXRlcih7XG4gICAgICAgIG1heFJlcXVlc3RzOiBNYXRoLmZsb29yKEdJVEhVQl9BUElfUkFURV9MSU1JVFMuVU5BVVRIRU5USUNBVEVEX0xJTUlUICogR0lUSFVCX0FQSV9SQVRFX0xJTUlUUy5CVUZGRVJfUEVSQ0VOVEFHRSksXG4gICAgICAgIHdpbmRvd01zOiBHSVRIVUJfQVBJX1JBVEVfTElNSVRTLldJTkRPV19NUyxcbiAgICAgICAgbWluRGVsYXlNczogR0lUSFVCX0FQSV9SQVRFX0xJTUlUUy5NSU5fREVMQVlfTVNcbiAgICAgIH0pO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBTZXR1cCBwZXJpb2RpYyBjaGVjayBmb3IgcmF0ZSBsaW1pdCBzdGF0dXNcbiAgICovXG4gIHByaXZhdGUgc2V0dXBQZXJpb2RpY1N0YXR1c0NoZWNrKCk6IHZvaWQge1xuICAgIC8vIENoZWNrIGF1dGggc3RhdHVzIGV2ZXJ5IDUgbWludXRlc1xuICAgIHNldEludGVydmFsKCgpID0+IHtcbiAgICAgIHRoaXMudXBkYXRlTGltaXRzRm9yQXV0aFN0YXR1cygpLmNhdGNoKGVycm9yID0+IHtcbiAgICAgICAgbG9nZ2VyLndhcm4oJ1BlcmlvZGljIGF1dGggc3RhdHVzIGNoZWNrIGZhaWxlZCcsIHsgZXJyb3IgfSk7XG4gICAgICB9KTtcbiAgICB9LCA1ICogNjAgKiAxMDAwKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBRdWV1ZSBhIEdpdEh1YiBBUEkgcmVxdWVzdCB3aXRoIHJhdGUgbGltaXRpbmdcbiAgICogQHBhcmFtIG9wZXJhdGlvbiBEZXNjcmlwdGlvbiBvZiB0aGUgb3BlcmF0aW9uXG4gICAqIEBwYXJhbSBhcGlDYWxsIEZ1bmN0aW9uIHRoYXQgbWFrZXMgdGhlIGFjdHVhbCBBUEkgY2FsbFxuICAgKiBAcGFyYW0gcHJpb3JpdHkgUmVxdWVzdCBwcmlvcml0eSAoaGlnaCwgbm9ybWFsLCBsb3cpXG4gICAqIEByZXR1cm5zIFByb21pc2UgdGhhdCByZXNvbHZlcyB3aXRoIHRoZSBBUEkgcmVzcG9uc2VcbiAgICovXG4gIGFzeW5jIHF1ZXVlUmVxdWVzdDxUPihcbiAgICBvcGVyYXRpb246IHN0cmluZyxcbiAgICBhcGlDYWxsOiAoKSA9PiBQcm9taXNlPFQ+LFxuICAgIHByaW9yaXR5OiAnaGlnaCcgfCAnbm9ybWFsJyB8ICdsb3cnID0gJ25vcm1hbCdcbiAgKTogUHJvbWlzZTxUPiB7XG4gICAgLy8gU0VDVVJJVFkgRklYIChETUNQLVNFQy0wMDQpOiBOb3JtYWxpemUgVW5pY29kZSBpbiBvcGVyYXRpb24gbmFtZSB0byBwcmV2ZW50IGluamVjdGlvbiBhdHRhY2tzXG4gICAgY29uc3Qgbm9ybWFsaXplZE9wZXJhdGlvbiA9IFVuaWNvZGVWYWxpZGF0b3Iubm9ybWFsaXplKG9wZXJhdGlvbik7XG4gICAgaWYgKCFub3JtYWxpemVkT3BlcmF0aW9uLmlzVmFsaWQpIHtcbiAgICAgIFNlY3VyaXR5TW9uaXRvci5sb2dTZWN1cml0eUV2ZW50KHtcbiAgICAgICAgdHlwZTogJ1VOSUNPREVfVkFMSURBVElPTl9FUlJPUicsXG4gICAgICAgIHNldmVyaXR5OiAnTUVESVVNJyxcbiAgICAgICAgc291cmNlOiAnR2l0SHViUmF0ZUxpbWl0ZXIucXVldWVSZXF1ZXN0JyxcbiAgICAgICAgZGV0YWlsczogYEludmFsaWQgVW5pY29kZSBpbiBvcGVyYXRpb24gbmFtZTogJHtub3JtYWxpemVkT3BlcmF0aW9uLmRldGVjdGVkSXNzdWVzPy5bMF0gfHwgJ3Vua25vd24gZXJyb3InfWBcbiAgICAgIH0pO1xuICAgICAgLy8gVXNlIGEgc2FmZSBmYWxsYmFjayBmb3IgdGhlIG9wZXJhdGlvbiBuYW1lXG4gICAgICBvcGVyYXRpb24gPSAnZ2l0aHViLWFwaS1yZXF1ZXN0JztcbiAgICB9IGVsc2Uge1xuICAgICAgb3BlcmF0aW9uID0gbm9ybWFsaXplZE9wZXJhdGlvbi5ub3JtYWxpemVkQ29udGVudDtcbiAgICB9XG4gICAgXG4gICAgY29uc3QgcmVxdWVzdElkID0gYCR7b3BlcmF0aW9ufS0ke0RhdGUubm93KCl9LSR7TWF0aC5yYW5kb20oKS50b1N0cmluZygzNikuc3Vic3RyKDIsIDkpfWA7XG4gICAgXG4gICAgcmV0dXJuIG5ldyBQcm9taXNlPFQ+KChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgIGNvbnN0IHJlcXVlc3Q6IEdpdEh1YkFwaVJlcXVlc3QgPSB7XG4gICAgICAgIGlkOiByZXF1ZXN0SWQsXG4gICAgICAgIG9wZXJhdGlvbixcbiAgICAgICAgcHJpb3JpdHksXG4gICAgICAgIHRpbWVzdGFtcDogRGF0ZS5ub3coKSxcbiAgICAgICAgcmVzb2x2ZTogYXN5bmMgKHZhbHVlKSA9PiB7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIGxvZ2dlci5kZWJ1ZygnRXhlY3V0aW5nIEdpdEh1YiBBUEkgcmVxdWVzdCcsIHtcbiAgICAgICAgICAgICAgb3BlcmF0aW9uLFxuICAgICAgICAgICAgICByZXF1ZXN0SWQsXG4gICAgICAgICAgICAgIHF1ZXVlV2FpdFRpbWU6IERhdGUubm93KCkgLSByZXF1ZXN0LnRpbWVzdGFtcFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBcbiAgICAgICAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IGFwaUNhbGwoKTtcbiAgICAgICAgICAgIHJlc29sdmUocmVzdWx0KTtcbiAgICAgICAgICAgIFxuICAgICAgICAgICAgLy8gTG9nIHN1Y2Nlc3NmdWwgQVBJIHVzYWdlIGZvciBxdW90YSB0cmFja2luZ1xuICAgICAgICAgICAgdGhpcy5sb2dBcGlVc2FnZShvcGVyYXRpb24sICdzdWNjZXNzJyk7XG4gICAgICAgICAgICBcbiAgICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgLy8gQ2hlY2sgaWYgdGhpcyBpcyBhIHJhdGUgbGltaXQgZXJyb3IgZnJvbSBHaXRIdWJcbiAgICAgICAgICAgIGlmICh0aGlzLmlzR2l0SHViUmF0ZUxpbWl0RXJyb3IoZXJyb3IpKSB7XG4gICAgICAgICAgICAgIHRoaXMuaGFuZGxlR2l0SHViUmF0ZUxpbWl0KGVycm9yKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJlamVjdChlcnJvcik7XG4gICAgICAgICAgICB0aGlzLmxvZ0FwaVVzYWdlKG9wZXJhdGlvbiwgJ2Vycm9yJywgZXJyb3IpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSxcbiAgICAgICAgcmVqZWN0XG4gICAgICB9O1xuXG4gICAgICAvLyBBZGQgdG8gcXVldWUgd2l0aCBwcmlvcml0eSBvcmRlcmluZ1xuICAgICAgdGhpcy5hZGRUb1F1ZXVlKHJlcXVlc3QpO1xuICAgICAgdGhpcy5wcm9jZXNzUXVldWUoKTtcbiAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBBZGQgcmVxdWVzdCB0byBxdWV1ZSB3aXRoIHByaW9yaXR5IG9yZGVyaW5nXG4gICAqL1xuICBwcml2YXRlIGFkZFRvUXVldWUocmVxdWVzdDogR2l0SHViQXBpUmVxdWVzdCk6IHZvaWQge1xuICAgIC8vIEluc2VydCBiYXNlZCBvbiBwcmlvcml0eTogaGlnaCA+IG5vcm1hbCA+IGxvd1xuICAgIC8vIFdpdGhpbiBzYW1lIHByaW9yaXR5LCBtYWludGFpbiBGSUZPIG9yZGVyXG4gICAgY29uc3QgcHJpb3JpdHlPcmRlciA9IHsgaGlnaDogMCwgbm9ybWFsOiAxLCBsb3c6IDIgfTtcbiAgICBcbiAgICBsZXQgaW5zZXJ0SW5kZXggPSB0aGlzLnJlcXVlc3RRdWV1ZS5sZW5ndGg7XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCB0aGlzLnJlcXVlc3RRdWV1ZS5sZW5ndGg7IGkrKykge1xuICAgICAgaWYgKHByaW9yaXR5T3JkZXJbcmVxdWVzdC5wcmlvcml0eV0gPCBwcmlvcml0eU9yZGVyW3RoaXMucmVxdWVzdFF1ZXVlW2ldLnByaW9yaXR5XSkge1xuICAgICAgICBpbnNlcnRJbmRleCA9IGk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgfVxuICAgIH1cbiAgICBcbiAgICB0aGlzLnJlcXVlc3RRdWV1ZS5zcGxpY2UoaW5zZXJ0SW5kZXgsIDAsIHJlcXVlc3QpO1xuICAgIFxuICAgIGxvZ2dlci5kZWJ1ZygnR2l0SHViIEFQSSByZXF1ZXN0IHF1ZXVlZCcsIHtcbiAgICAgIG9wZXJhdGlvbjogcmVxdWVzdC5vcGVyYXRpb24sXG4gICAgICBwcmlvcml0eTogcmVxdWVzdC5wcmlvcml0eSxcbiAgICAgIHF1ZXVlUG9zaXRpb246IGluc2VydEluZGV4LFxuICAgICAgdG90YWxRdWV1ZWQ6IHRoaXMucmVxdWVzdFF1ZXVlLmxlbmd0aFxuICAgIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIFByb2Nlc3MgdGhlIHJlcXVlc3QgcXVldWVcbiAgICovXG4gIHByaXZhdGUgYXN5bmMgcHJvY2Vzc1F1ZXVlKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGlmICh0aGlzLnByb2Nlc3NpbmcgfHwgdGhpcy5yZXF1ZXN0UXVldWUubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgdGhpcy5wcm9jZXNzaW5nID0gdHJ1ZTtcblxuICAgIHRyeSB7XG4gICAgICB3aGlsZSAodGhpcy5yZXF1ZXN0UXVldWUubGVuZ3RoID4gMCkge1xuICAgICAgICAvLyBVcGRhdGUgYXV0aCBzdGF0dXMgcGVyaW9kaWNhbGx5XG4gICAgICAgIGlmIChNYXRoLnJhbmRvbSgpIDwgMC4xKSB7IC8vIDEwJSBjaGFuY2VcbiAgICAgICAgICBhd2FpdCB0aGlzLnVwZGF0ZUxpbWl0c0ZvckF1dGhTdGF0dXMoKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IHJhdGVMaW1pdFN0YXR1cyA9IHRoaXMucmF0ZUxpbWl0ZXIuY2hlY2tMaW1pdCgpO1xuICAgICAgICBcbiAgICAgICAgaWYgKCFyYXRlTGltaXRTdGF0dXMuYWxsb3dlZCkge1xuICAgICAgICAgIC8vIExvZyByYXRlIGxpbWl0IHdhaXRcbiAgICAgICAgICBsb2dnZXIuaW5mbygnR2l0SHViIEFQSSByYXRlIGxpbWl0IHJlYWNoZWQsIHdhaXRpbmcnLCB7XG4gICAgICAgICAgICByZXRyeUFmdGVyTXM6IHJhdGVMaW1pdFN0YXR1cy5yZXRyeUFmdGVyTXMsXG4gICAgICAgICAgICByZW1haW5pbmdUb2tlbnM6IHJhdGVMaW1pdFN0YXR1cy5yZW1haW5pbmdUb2tlbnMsXG4gICAgICAgICAgICBxdWV1ZUxlbmd0aDogdGhpcy5yZXF1ZXN0UXVldWUubGVuZ3RoLFxuICAgICAgICAgICAgcmVzZXRUaW1lOiByYXRlTGltaXRTdGF0dXMucmVzZXRUaW1lXG4gICAgICAgICAgfSk7XG5cbiAgICAgICAgICAvLyBXYWl0IGZvciB0aGUgc3BlY2lmaWVkIHRpbWVcbiAgICAgICAgICBhd2FpdCBuZXcgUHJvbWlzZShyZXNvbHZlID0+IHNldFRpbWVvdXQocmVzb2x2ZSwgcmF0ZUxpbWl0U3RhdHVzLnJldHJ5QWZ0ZXJNcyB8fCAxMDAwKSk7XG4gICAgICAgICAgY29udGludWU7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBQcm9jZXNzIHRoZSBuZXh0IHJlcXVlc3RcbiAgICAgICAgY29uc3QgcmVxdWVzdCA9IHRoaXMucmVxdWVzdFF1ZXVlLnNoaWZ0KCkhO1xuICAgICAgICB0aGlzLnJhdGVMaW1pdGVyLmNvbnN1bWVUb2tlbigpO1xuXG4gICAgICAgIC8vIEV4ZWN1dGUgdGhlIHJlcXVlc3RcbiAgICAgICAgcmVxdWVzdC5yZXNvbHZlKG51bGwpOyAvLyBUaGlzIHdpbGwgdHJpZ2dlciB0aGUgYWN0dWFsIEFQSSBjYWxsXG4gICAgICB9XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIHRoaXMucHJvY2Vzc2luZyA9IGZhbHNlO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgY3VycmVudCByYXRlIGxpbWl0IHN0YXR1c1xuICAgKi9cbiAgZ2V0U3RhdHVzKCk6IEdpdEh1YlJhdGVTdGF0dXMge1xuICAgIGNvbnN0IGJhc2VTdGF0dXMgPSB0aGlzLnJhdGVMaW1pdGVyLmdldFN0YXR1cygpO1xuICAgIFxuICAgIHJldHVybiB7XG4gICAgICAuLi5iYXNlU3RhdHVzLFxuICAgICAgcXVldWVMZW5ndGg6IHRoaXMucmVxdWVzdFF1ZXVlLmxlbmd0aCxcbiAgICAgIGN1cnJlbnRMaW1pdDogdGhpcy5pc0F1dGhlbnRpY2F0ZWQgXG4gICAgICAgID8gR0lUSFVCX0FQSV9SQVRFX0xJTUlUUy5BVVRIRU5USUNBVEVEX0xJTUlUIFxuICAgICAgICA6IEdJVEhVQl9BUElfUkFURV9MSU1JVFMuVU5BVVRIRU5USUNBVEVEX0xJTUlULFxuICAgICAgcmF0ZUxpbWl0SW5mbzogdGhpcy5sYXN0UmF0ZUxpbWl0SW5mb1xuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogQ2hlY2sgaWYgYW4gZXJyb3IgaXMgYSBHaXRIdWIgcmF0ZSBsaW1pdCBlcnJvclxuICAgKi9cbiAgcHJpdmF0ZSBpc0dpdEh1YlJhdGVMaW1pdEVycm9yKGVycm9yOiBhbnkpOiBib29sZWFuIHtcbiAgICByZXR1cm4gZXJyb3I/LnN0YXR1cyA9PT0gNDI5IHx8IFxuICAgICAgICAgICBlcnJvcj8ucmVzcG9uc2U/LnN0YXR1cyA9PT0gNDI5IHx8XG4gICAgICAgICAgICh0eXBlb2YgZXJyb3I/Lm1lc3NhZ2UgPT09ICdzdHJpbmcnICYmIGVycm9yLm1lc3NhZ2UudG9Mb3dlckNhc2UoKS5pbmNsdWRlcygncmF0ZSBsaW1pdCcpKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBIYW5kbGUgR2l0SHViIHJhdGUgbGltaXQgZXJyb3IgcmVzcG9uc2VcbiAgICovXG4gIHByaXZhdGUgaGFuZGxlR2l0SHViUmF0ZUxpbWl0KGVycm9yOiBhbnkpOiB2b2lkIHtcbiAgICBsZXQgcmVzZXRUaW1lOiBEYXRlIHwgdW5kZWZpbmVkO1xuICAgIGxldCByZW1haW5pbmdSZXF1ZXN0cyA9IDA7XG4gICAgXG4gICAgLy8gUGFyc2UgcmF0ZSBsaW1pdCBoZWFkZXJzIGlmIGF2YWlsYWJsZVxuICAgIGlmIChlcnJvcj8ucmVzcG9uc2U/LmhlYWRlcnMpIHtcbiAgICAgIGNvbnN0IGhlYWRlcnMgPSBlcnJvci5yZXNwb25zZS5oZWFkZXJzO1xuICAgICAgY29uc3QgcmVzZXRUaW1lc3RhbXAgPSBwYXJzZUludChoZWFkZXJzWyd4LXJhdGVsaW1pdC1yZXNldCddIHx8ICcwJyk7XG4gICAgICBjb25zdCByZW1haW5pbmcgPSBwYXJzZUludChoZWFkZXJzWyd4LXJhdGVsaW1pdC1yZW1haW5pbmcnXSB8fCAnMCcpO1xuICAgICAgY29uc3QgbGltaXQgPSBwYXJzZUludChoZWFkZXJzWyd4LXJhdGVsaW1pdC1saW1pdCddIHx8ICcwJyk7XG4gICAgICBcbiAgICAgIGlmIChyZXNldFRpbWVzdGFtcCA+IDApIHtcbiAgICAgICAgcmVzZXRUaW1lID0gbmV3IERhdGUocmVzZXRUaW1lc3RhbXAgKiAxMDAwKTtcbiAgICAgIH1cbiAgICAgIFxuICAgICAgdGhpcy5sYXN0UmF0ZUxpbWl0SW5mbyA9IHtcbiAgICAgICAgbGltaXQsXG4gICAgICAgIHJlbWFpbmluZyxcbiAgICAgICAgcmVzZXQ6IHJlc2V0VGltZSB8fCBuZXcgRGF0ZShEYXRlLm5vdygpICsgNjAgKiA2MCAqIDEwMDApLCAvLyBEZWZhdWx0IHRvIDEgaG91clxuICAgICAgICB1c2VkOiBsaW1pdCAtIHJlbWFpbmluZ1xuICAgICAgfTtcbiAgICAgIFxuICAgICAgcmVtYWluaW5nUmVxdWVzdHMgPSByZW1haW5pbmc7XG4gICAgfVxuXG4gICAgbG9nZ2VyLndhcm4oJ0dpdEh1YiBBUEkgcmF0ZSBsaW1pdCBoaXQgZnJvbSBzZXJ2ZXInLCB7XG4gICAgICByZW1haW5pbmc6IHJlbWFpbmluZ1JlcXVlc3RzLFxuICAgICAgcmVzZXRUaW1lLFxuICAgICAgcXVldWVMZW5ndGg6IHRoaXMucmVxdWVzdFF1ZXVlLmxlbmd0aCxcbiAgICAgIGVycm9yTWVzc2FnZTogZXJyb3I/Lm1lc3NhZ2VcbiAgICB9KTtcblxuICAgIC8vIExvZyBhcyBhIHNlY3VyaXR5IGV2ZW50IGZvciBtb25pdG9yaW5nXG4gICAgU2VjdXJpdHlNb25pdG9yLmxvZ1NlY3VyaXR5RXZlbnQoe1xuICAgICAgdHlwZTogJ1JBVEVfTElNSVRfRVhDRUVERUQnLFxuICAgICAgc2V2ZXJpdHk6ICdNRURJVU0nLFxuICAgICAgc291cmNlOiAnR2l0SHViUmF0ZUxpbWl0ZXIuaGFuZGxlR2l0SHViUmF0ZUxpbWl0JyxcbiAgICAgIGRldGFpbHM6IGBHaXRIdWIgQVBJIHJhdGUgbGltaXQgZXhjZWVkZWQuIFJlbWFpbmluZzogJHtyZW1haW5pbmdSZXF1ZXN0c30sIFF1ZXVlOiAke3RoaXMucmVxdWVzdFF1ZXVlLmxlbmd0aH1gLFxuICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgcmF0ZUxpbWl0SW5mbzogdGhpcy5sYXN0UmF0ZUxpbWl0SW5mbyxcbiAgICAgICAgYXV0aGVudGljYXRlZDogdGhpcy5pc0F1dGhlbnRpY2F0ZWRcbiAgICAgIH1cbiAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBMb2cgQVBJIHVzYWdlIGZvciBtb25pdG9yaW5nIGFuZCBkaWFnbm9zdGljc1xuICAgKi9cbiAgcHJpdmF0ZSBsb2dBcGlVc2FnZShvcGVyYXRpb246IHN0cmluZywgcmVzdWx0OiAnc3VjY2VzcycgfCAnZXJyb3InLCBlcnJvcj86IGFueSk6IHZvaWQge1xuICAgIGNvbnN0IHN0YXR1cyA9IHRoaXMuZ2V0U3RhdHVzKCk7XG4gICAgXG4gICAgbG9nZ2VyLmRlYnVnKCdHaXRIdWIgQVBJIHVzYWdlIGxvZ2dlZCcsIHtcbiAgICAgIG9wZXJhdGlvbixcbiAgICAgIHJlc3VsdCxcbiAgICAgIHJlbWFpbmluZ1Rva2Vuczogc3RhdHVzLnJlbWFpbmluZ1Rva2VucyxcbiAgICAgIHF1ZXVlTGVuZ3RoOiBzdGF0dXMucXVldWVMZW5ndGgsXG4gICAgICBhdXRoZW50aWNhdGVkOiB0aGlzLmlzQXV0aGVudGljYXRlZCxcbiAgICAgIGVycm9yOiBlcnJvcj8ubWVzc2FnZVxuICAgIH0pO1xuXG4gICAgLy8gTG9nIHdhcm5pbmcgaWYgZ2V0dGluZyBjbG9zZSB0byByYXRlIGxpbWl0c1xuICAgIGlmIChzdGF0dXMucmVtYWluaW5nVG9rZW5zIDwgMTAwICYmIHRoaXMuaXNBdXRoZW50aWNhdGVkKSB7XG4gICAgICBsb2dnZXIud2FybignQXBwcm9hY2hpbmcgR2l0SHViIEFQSSByYXRlIGxpbWl0Jywge1xuICAgICAgICBvcGVyYXRpb24sXG4gICAgICAgIHJlbWFpbmluZ1Rva2Vuczogc3RhdHVzLnJlbWFpbmluZ1Rva2VucyxcbiAgICAgICAgY3VycmVudExpbWl0OiBzdGF0dXMuY3VycmVudExpbWl0LFxuICAgICAgICByZWNvbW1lbmRhdGlvbjogJ0NvbnNpZGVyIHJlZHVjaW5nIEFQSSB1c2FnZSBmcmVxdWVuY3knXG4gICAgICB9KTtcbiAgICB9IGVsc2UgaWYgKHN0YXR1cy5yZW1haW5pbmdUb2tlbnMgPCAxMCAmJiAhdGhpcy5pc0F1dGhlbnRpY2F0ZWQpIHtcbiAgICAgIGxvZ2dlci53YXJuKCdBcHByb2FjaGluZyBHaXRIdWIgQVBJIHJhdGUgbGltaXQgKHVuYXV0aGVudGljYXRlZCknLCB7XG4gICAgICAgIG9wZXJhdGlvbixcbiAgICAgICAgcmVtYWluaW5nVG9rZW5zOiBzdGF0dXMucmVtYWluaW5nVG9rZW5zLFxuICAgICAgICBjdXJyZW50TGltaXQ6IHN0YXR1cy5jdXJyZW50TGltaXQsXG4gICAgICAgIHJlY29tbWVuZGF0aW9uOiAnQ29uc2lkZXIgYXV0aGVudGljYXRpbmcgZm9yIGhpZ2hlciByYXRlIGxpbWl0cydcbiAgICAgIH0pO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBDbGVhciB0aGUgcmVxdWVzdCBxdWV1ZSAoZm9yIHRlc3Rpbmcgb3IgZW1lcmdlbmN5IHNpdHVhdGlvbnMpXG4gICAqL1xuICBjbGVhclF1ZXVlKCk6IHZvaWQge1xuICAgIGNvbnN0IGNsZWFyZWRDb3VudCA9IHRoaXMucmVxdWVzdFF1ZXVlLmxlbmd0aDtcbiAgICBcbiAgICAvLyBSZWplY3QgYWxsIHBlbmRpbmcgcmVxdWVzdHNcbiAgICB0aGlzLnJlcXVlc3RRdWV1ZS5mb3JFYWNoKHJlcXVlc3QgPT4ge1xuICAgICAgcmVxdWVzdC5yZWplY3QobmV3IEVycm9yKCdSZXF1ZXN0IHF1ZXVlIGNsZWFyZWQnKSk7XG4gICAgfSk7XG4gICAgXG4gICAgdGhpcy5yZXF1ZXN0UXVldWUgPSBbXTtcbiAgICBcbiAgICBsb2dnZXIuaW5mbygnR2l0SHViIEFQSSByZXF1ZXN0IHF1ZXVlIGNsZWFyZWQnLCB7IGNsZWFyZWRDb3VudCB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXNldCB0aGUgcmF0ZSBsaW1pdGVyIChmb3IgdGVzdGluZylcbiAgICovXG4gIHJlc2V0KCk6IHZvaWQge1xuICAgIHRoaXMucmF0ZUxpbWl0ZXIucmVzZXQoKTtcbiAgICB0aGlzLmNsZWFyUXVldWUoKTtcbiAgICB0aGlzLnByb2Nlc3NpbmcgPSBmYWxzZTtcbiAgICBsb2dnZXIuaW5mbygnR2l0SHViIHJhdGUgbGltaXRlciByZXNldCcpO1xuICB9XG59XG5cbi8vIFNpbmdsZXRvbiBpbnN0YW5jZSBmb3IgZ2xvYmFsIHVzZVxuZXhwb3J0IGNvbnN0IGdpdGh1YlJhdGVMaW1pdGVyID0gbmV3IEdpdEh1YlJhdGVMaW1pdGVyKCk7Il19
@@ -0,0 +1,134 @@
1
+ /**
2
+ * Comprehensive Performance Monitoring System
3
+ * Tracks search times, memory usage, cache performance, and system metrics
4
+ */
5
+ export interface PerformanceMetrics {
6
+ searchTimes: number[];
7
+ memoryUsage: MemoryUsage[];
8
+ cacheStats: CachePerformance;
9
+ systemStats: SystemStats;
10
+ timestamp: Date;
11
+ }
12
+ export interface MemoryUsage {
13
+ heapUsed: number;
14
+ heapTotal: number;
15
+ rss: number;
16
+ external: number;
17
+ timestamp: Date;
18
+ }
19
+ export interface CachePerformance {
20
+ hitRate: number;
21
+ avgHitTime: number;
22
+ avgMissTime: number;
23
+ totalHits: number;
24
+ totalMisses: number;
25
+ evictions: number;
26
+ }
27
+ export interface SystemStats {
28
+ cpuUsage: number;
29
+ loadAverage: number[];
30
+ freeMemory: number;
31
+ totalMemory: number;
32
+ uptime: number;
33
+ }
34
+ export interface SearchMetrics {
35
+ query: string;
36
+ duration: number;
37
+ resultCount: number;
38
+ sources: string[];
39
+ cacheHit: boolean;
40
+ memoryBefore: number;
41
+ memoryAfter: number;
42
+ timestamp: Date;
43
+ }
44
+ export interface SlowQuery {
45
+ query: string;
46
+ duration: number;
47
+ threshold: number;
48
+ sources: string[];
49
+ resultCount: number;
50
+ memoryUsage: number;
51
+ timestamp: Date;
52
+ }
53
+ export declare class PerformanceMonitor {
54
+ private static instance;
55
+ private searchMetrics;
56
+ private slowQueries;
57
+ private memorySnapshots;
58
+ private cacheMetrics;
59
+ private readonly maxMetricsHistory;
60
+ private readonly slowQueryThreshold;
61
+ private readonly memorySnapshotInterval;
62
+ private readonly maxSlowQueries;
63
+ private memoryMonitorInterval?;
64
+ private isMonitoring;
65
+ private constructor();
66
+ static getInstance(): PerformanceMonitor;
67
+ /**
68
+ * Start performance monitoring
69
+ */
70
+ startMonitoring(): void;
71
+ /**
72
+ * Stop performance monitoring
73
+ */
74
+ stopMonitoring(): void;
75
+ /**
76
+ * Record search performance metrics
77
+ */
78
+ recordSearch(metrics: SearchMetrics): void;
79
+ /**
80
+ * Record cache performance metrics
81
+ */
82
+ recordCachePerformance(cacheName: string, stats: CachePerformance): void;
83
+ /**
84
+ * Get comprehensive performance metrics
85
+ */
86
+ getMetrics(): PerformanceMetrics;
87
+ /**
88
+ * Get search performance statistics
89
+ */
90
+ getSearchStats(): {
91
+ totalSearches: number;
92
+ averageTime: number;
93
+ medianTime: number;
94
+ p95Time: number;
95
+ p99Time: number;
96
+ slowQueries: number;
97
+ cacheHitRate: number;
98
+ };
99
+ /**
100
+ * Get memory usage statistics
101
+ */
102
+ getMemoryStats(): {
103
+ currentUsage: MemoryUsage;
104
+ peakUsage: MemoryUsage;
105
+ averageUsage: MemoryUsage;
106
+ growthRate: number;
107
+ };
108
+ /**
109
+ * Get slow queries with analysis
110
+ */
111
+ getSlowQueries(limit?: number): SlowQuery[];
112
+ /**
113
+ * Analyze performance trends
114
+ */
115
+ analyzeTrends(): {
116
+ performanceTrend: 'improving' | 'degrading' | 'stable';
117
+ memoryTrend: 'growing' | 'shrinking' | 'stable';
118
+ recommendations: string[];
119
+ };
120
+ /**
121
+ * Reset all performance metrics
122
+ */
123
+ reset(): void;
124
+ /**
125
+ * Export metrics for external analysis
126
+ */
127
+ exportMetrics(): string;
128
+ private recordSlowQuery;
129
+ private startMemoryMonitoring;
130
+ private takeMemorySnapshot;
131
+ private aggregateCacheStats;
132
+ private getSystemStats;
133
+ }
134
+ //# sourceMappingURL=PerformanceMonitor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PerformanceMonitor.d.ts","sourceRoot":"","sources":["../../src/utils/PerformanceMonitor.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAKH,MAAM,WAAW,kBAAkB;IACjC,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,WAAW,EAAE,WAAW,EAAE,CAAC;IAC3B,UAAU,EAAE,gBAAgB,CAAC;IAC7B,WAAW,EAAE,WAAW,CAAC;IACzB,SAAS,EAAE,IAAI,CAAC;CACjB;AAED,MAAM,WAAW,WAAW;IAC1B,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,GAAG,EAAE,MAAM,CAAC;IACZ,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,IAAI,CAAC;CACjB;AAED,MAAM,WAAW,gBAAgB;IAC/B,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,WAAW;IAC1B,QAAQ,EAAE,MAAM,CAAC;IACjB,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;IACpB,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,aAAa;IAC5B,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,WAAW,EAAE,MAAM,CAAC;IACpB,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,QAAQ,EAAE,OAAO,CAAC;IAClB,YAAY,EAAE,MAAM,CAAC;IACrB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,IAAI,CAAC;CACjB;AAED,MAAM,WAAW,SAAS;IACxB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,IAAI,CAAC;CACjB;AAED,qBAAa,kBAAkB;IAC7B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAmC;IAE1D,OAAO,CAAC,aAAa,CAAuB;IAC5C,OAAO,CAAC,WAAW,CAAmB;IACtC,OAAO,CAAC,eAAe,CAAqB;IAC5C,OAAO,CAAC,YAAY,CAA4C;IAGhE,OAAO,CAAC,QAAQ,CAAC,iBAAiB,CAAQ;IAC1C,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAAO;IAC1C,OAAO,CAAC,QAAQ,CAAC,sBAAsB,CAAS;IAChD,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAO;IAGtC,OAAO,CAAC,qBAAqB,CAAC,CAAiB;IAC/C,OAAO,CAAC,YAAY,CAAS;IAE7B,OAAO;WAIO,WAAW,IAAI,kBAAkB;IAO/C;;OAEG;IACH,eAAe,IAAI,IAAI;IAcvB;;OAEG;IACH,cAAc,IAAI,IAAI;IAWtB;;OAEG;IACH,YAAY,CAAC,OAAO,EAAE,aAAa,GAAG,IAAI;IA2C1C;;OAEG;IACH,sBAAsB,CAAC,SAAS,EAAE,MAAM,EAAE,KAAK,EAAE,gBAAgB,GAAG,IAAI;IAqBxE;;OAEG;IACH,UAAU,IAAI,kBAAkB;IAUhC;;OAEG;IACH,cAAc,IAAI;QAChB,aAAa,EAAE,MAAM,CAAC;QACtB,WAAW,EAAE,MAAM,CAAC;QACpB,UAAU,EAAE,MAAM,CAAC;QACnB,OAAO,EAAE,MAAM,CAAC;QAChB,OAAO,EAAE,MAAM,CAAC;QAChB,WAAW,EAAE,MAAM,CAAC;QACpB,YAAY,EAAE,MAAM,CAAC;KACtB;IA2BD;;OAEG;IACH,cAAc,IAAI;QAChB,YAAY,EAAE,WAAW,CAAC;QAC1B,SAAS,EAAE,WAAW,CAAC;QACvB,YAAY,EAAE,WAAW,CAAC;QAC1B,UAAU,EAAE,MAAM,CAAC;KACpB;IA2CD;;OAEG;IACH,cAAc,CAAC,KAAK,GAAE,MAAW,GAAG,SAAS,EAAE;IAM/C;;OAEG;IACH,aAAa,IAAI;QACf,gBAAgB,EAAE,WAAW,GAAG,WAAW,GAAG,QAAQ,CAAC;QACvD,WAAW,EAAE,SAAS,GAAG,WAAW,GAAG,QAAQ,CAAC;QAChD,eAAe,EAAE,MAAM,EAAE,CAAC;KAC3B;IAmDD;;OAEG;IACH,KAAK,IAAI,IAAI;IASb;;OAEG;IACH,aAAa,IAAI,MAAM;IAcvB,OAAO,CAAC,eAAe;IASvB,OAAO,CAAC,qBAAqB;IAkB7B,OAAO,CAAC,kBAAkB;IAW1B,OAAO,CAAC,mBAAmB;IAoC3B,OAAO,CAAC,cAAc;CAWvB"}