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