@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,{"version":3,"file":"GitHubRateLimiter.js","sourceRoot":"","sources":["../../src/utils/GitHubRateLimiter.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,EAAE,WAAW,EAAsC,MAAM,kBAAkB,CAAC;AACnF,OAAO,EAAE,sBAAsB,EAAE,MAAM,kCAAkC,CAAC;AAC1E,OAAO,EAAE,YAAY,EAAE,MAAM,6BAA6B,CAAC;AAC3D,OAAO,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AACrC,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AACjE,OAAO,EAAE,gBAAgB,EAAE,MAAM,4CAA4C,CAAC;AAwB9E,MAAM,OAAO,iBAAiB;IACpB,WAAW,CAAe;IAC1B,YAAY,GAAuB,EAAE,CAAC;IACtC,UAAU,GAAG,KAAK,CAAC;IACnB,iBAAiB,CAAuB;IACxC,eAAe,GAAG,KAAK,CAAC;IAEhC;QACE,yEAAyE;QACzE,IAAI,CAAC,yBAAyB,EAAE,CAAC;QACjC,IAAI,CAAC,wBAAwB,EAAE,CAAC;IAClC,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,yBAAyB;QACrC,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,MAAM,YAAY,CAAC,mBAAmB,EAAE,CAAC;YACvD,MAAM,kBAAkB,GAAG,CAAC,CAAC,KAAK,CAAC;YAEnC,oDAAoD;YACpD,IAAI,kBAAkB,KAAK,IAAI,CAAC,eAAe,EAAE,CAAC;gBAChD,IAAI,CAAC,eAAe,GAAG,kBAAkB,CAAC;gBAE1C,MAAM,KAAK,GAAG,IAAI,CAAC,eAAe;oBAChC,CAAC,CAAC,sBAAsB,CAAC,mBAAmB;oBAC5C,CAAC,CAAC,sBAAsB,CAAC,qBAAqB,CAAC;gBAEjD,2CAA2C;gBAC3C,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,sBAAsB,CAAC,iBAAiB,CAAC,CAAC;gBAEnF,MAAM,MAAM,GAAsB;oBAChC,WAAW,EAAE,aAAa;oBAC1B,QAAQ,EAAE,sBAAsB,CAAC,SAAS;oBAC1C,UAAU,EAAE,sBAAsB,CAAC,YAAY;iBAChD,CAAC;gBAEF,IAAI,CAAC,WAAW,GAAG,IAAI,WAAW,CAAC,MAAM,CAAC,CAAC;gBAE3C,MAAM,CAAC,IAAI,CAAC,6BAA6B,EAAE;oBACzC,aAAa,EAAE,IAAI,CAAC,eAAe;oBACnC,KAAK,EAAE,aAAa;oBACpB,aAAa,EAAE,KAAK;oBACpB,gBAAgB,EAAE,sBAAsB,CAAC,iBAAiB;iBAC3D,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,IAAI,CAAC,yDAAyD,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC;YAClF,sCAAsC;YACtC,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;YAC7B,IAAI,CAAC,WAAW,GAAG,IAAI,WAAW,CAAC;gBACjC,WAAW,EAAE,IAAI,CAAC,KAAK,CAAC,sBAAsB,CAAC,qBAAqB,GAAG,sBAAsB,CAAC,iBAAiB,CAAC;gBAChH,QAAQ,EAAE,sBAAsB,CAAC,SAAS;gBAC1C,UAAU,EAAE,sBAAsB,CAAC,YAAY;aAChD,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED;;OAEG;IACK,wBAAwB;QAC9B,oCAAoC;QACpC,WAAW,CAAC,GAAG,EAAE;YACf,IAAI,CAAC,yBAAyB,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE;gBAC7C,MAAM,CAAC,IAAI,CAAC,mCAAmC,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC;YAC9D,CAAC,CAAC,CAAC;QACL,CAAC,EAAE,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC;IACpB,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,YAAY,CAChB,SAAiB,EACjB,OAAyB,EACzB,WAAsC,QAAQ;QAE9C,gGAAgG;QAChG,MAAM,mBAAmB,GAAG,gBAAgB,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;QAClE,IAAI,CAAC,mBAAmB,CAAC,OAAO,EAAE,CAAC;YACjC,eAAe,CAAC,gBAAgB,CAAC;gBAC/B,IAAI,EAAE,0BAA0B;gBAChC,QAAQ,EAAE,QAAQ;gBAClB,MAAM,EAAE,gCAAgC;gBACxC,OAAO,EAAE,sCAAsC,mBAAmB,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,IAAI,eAAe,EAAE;aAC5G,CAAC,CAAC;YACH,6CAA6C;YAC7C,SAAS,GAAG,oBAAoB,CAAC;QACnC,CAAC;aAAM,CAAC;YACN,SAAS,GAAG,mBAAmB,CAAC,iBAAiB,CAAC;QACpD,CAAC;QAED,MAAM,SAAS,GAAG,GAAG,SAAS,IAAI,IAAI,CAAC,GAAG,EAAE,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC;QAE1F,OAAO,IAAI,OAAO,CAAI,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACxC,MAAM,OAAO,GAAqB;gBAChC,EAAE,EAAE,SAAS;gBACb,SAAS;gBACT,QAAQ;gBACR,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;gBACrB,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,EAAE;oBACvB,IAAI,CAAC;wBACH,MAAM,CAAC,KAAK,CAAC,8BAA8B,EAAE;4BAC3C,SAAS;4BACT,SAAS;4BACT,aAAa,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO,CAAC,SAAS;yBAC9C,CAAC,CAAC;wBAEH,MAAM,MAAM,GAAG,MAAM,OAAO,EAAE,CAAC;wBAC/B,OAAO,CAAC,MAAM,CAAC,CAAC;wBAEhB,8CAA8C;wBAC9C,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;oBAEzC,CAAC;oBAAC,OAAO,KAAK,EAAE,CAAC;wBACf,kDAAkD;wBAClD,IAAI,IAAI,CAAC,sBAAsB,CAAC,KAAK,CAAC,EAAE,CAAC;4BACvC,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;wBACpC,CAAC;wBACD,MAAM,CAAC,KAAK,CAAC,CAAC;wBACd,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,OAAO,EAAE,KAAK,CAAC,CAAC;oBAC9C,CAAC;gBACH,CAAC;gBACD,MAAM;aACP,CAAC;YAEF,sCAAsC;YACtC,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;YACzB,IAAI,CAAC,YAAY,EAAE,CAAC;QACtB,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,UAAU,CAAC,OAAyB;QAC1C,gDAAgD;QAChD,4CAA4C;QAC5C,MAAM,aAAa,GAAG,EAAE,IAAI,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE,CAAC;QAErD,IAAI,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC;QAC3C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YAClD,IAAI,aAAa,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAG,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,EAAE,CAAC;gBACnF,WAAW,GAAG,CAAC,CAAC;gBAChB,MAAM;YACR,CAAC;QACH,CAAC;QAED,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC,EAAE,OAAO,CAAC,CAAC;QAElD,MAAM,CAAC,KAAK,CAAC,2BAA2B,EAAE;YACxC,SAAS,EAAE,OAAO,CAAC,SAAS;YAC5B,QAAQ,EAAE,OAAO,CAAC,QAAQ;YAC1B,aAAa,EAAE,WAAW;YAC1B,WAAW,EAAE,IAAI,CAAC,YAAY,CAAC,MAAM;SACtC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,YAAY;QACxB,IAAI,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACtD,OAAO;QACT,CAAC;QAED,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QAEvB,IAAI,CAAC;YACH,OAAO,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACpC,kCAAkC;gBAClC,IAAI,IAAI,CAAC,MAAM,EAAE,GAAG,GAAG,EAAE,CAAC,CAAC,aAAa;oBACtC,MAAM,IAAI,CAAC,yBAAyB,EAAE,CAAC;gBACzC,CAAC;gBAED,MAAM,eAAe,GAAG,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,CAAC;gBAEtD,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,CAAC;oBAC7B,sBAAsB;oBACtB,MAAM,CAAC,IAAI,CAAC,wCAAwC,EAAE;wBACpD,YAAY,EAAE,eAAe,CAAC,YAAY;wBAC1C,eAAe,EAAE,eAAe,CAAC,eAAe;wBAChD,WAAW,EAAE,IAAI,CAAC,YAAY,CAAC,MAAM;wBACrC,SAAS,EAAE,eAAe,CAAC,SAAS;qBACrC,CAAC,CAAC;oBAEH,8BAA8B;oBAC9B,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,eAAe,CAAC,YAAY,IAAI,IAAI,CAAC,CAAC,CAAC;oBACxF,SAAS;gBACX,CAAC;gBAED,2BAA2B;gBAC3B,MAAM,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,EAAG,CAAC;gBAC3C,IAAI,CAAC,WAAW,CAAC,YAAY,EAAE,CAAC;gBAEhC,sBAAsB;gBACtB,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,wCAAwC;YACjE,CAAC;QACH,CAAC;gBAAS,CAAC;YACT,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;QAC1B,CAAC;IACH,CAAC;IAED;;OAEG;IACH,SAAS;QACP,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC;QAEhD,OAAO;YACL,GAAG,UAAU;YACb,WAAW,EAAE,IAAI,CAAC,YAAY,CAAC,MAAM;YACrC,YAAY,EAAE,IAAI,CAAC,eAAe;gBAChC,CAAC,CAAC,sBAAsB,CAAC,mBAAmB;gBAC5C,CAAC,CAAC,sBAAsB,CAAC,qBAAqB;YAChD,aAAa,EAAE,IAAI,CAAC,iBAAiB;SACtC,CAAC;IACJ,CAAC;IAED;;OAEG;IACK,sBAAsB,CAAC,KAAU;QACvC,OAAO,KAAK,EAAE,MAAM,KAAK,GAAG;YACrB,KAAK,EAAE,QAAQ,EAAE,MAAM,KAAK,GAAG;YAC/B,CAAC,OAAO,KAAK,EAAE,OAAO,KAAK,QAAQ,IAAI,KAAK,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC;IACpG,CAAC;IAED;;OAEG;IACK,qBAAqB,CAAC,KAAU;QACtC,IAAI,SAA2B,CAAC;QAChC,IAAI,iBAAiB,GAAG,CAAC,CAAC;QAE1B,wCAAwC;QACxC,IAAI,KAAK,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC;YAC7B,MAAM,OAAO,GAAG,KAAK,CAAC,QAAQ,CAAC,OAAO,CAAC;YACvC,MAAM,cAAc,GAAG,QAAQ,CAAC,OAAO,CAAC,mBAAmB,CAAC,IAAI,GAAG,CAAC,CAAC;YACrE,MAAM,SAAS,GAAG,QAAQ,CAAC,OAAO,CAAC,uBAAuB,CAAC,IAAI,GAAG,CAAC,CAAC;YACpE,MAAM,KAAK,GAAG,QAAQ,CAAC,OAAO,CAAC,mBAAmB,CAAC,IAAI,GAAG,CAAC,CAAC;YAE5D,IAAI,cAAc,GAAG,CAAC,EAAE,CAAC;gBACvB,SAAS,GAAG,IAAI,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,CAAC;YAC9C,CAAC;YAED,IAAI,CAAC,iBAAiB,GAAG;gBACvB,KAAK;gBACL,SAAS;gBACT,KAAK,EAAE,SAAS,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,EAAE,oBAAoB;gBAC/E,IAAI,EAAE,KAAK,GAAG,SAAS;aACxB,CAAC;YAEF,iBAAiB,GAAG,SAAS,CAAC;QAChC,CAAC;QAED,MAAM,CAAC,IAAI,CAAC,uCAAuC,EAAE;YACnD,SAAS,EAAE,iBAAiB;YAC5B,SAAS;YACT,WAAW,EAAE,IAAI,CAAC,YAAY,CAAC,MAAM;YACrC,YAAY,EAAE,KAAK,EAAE,OAAO;SAC7B,CAAC,CAAC;QAEH,yCAAyC;QACzC,eAAe,CAAC,gBAAgB,CAAC;YAC/B,IAAI,EAAE,qBAAqB;YAC3B,QAAQ,EAAE,QAAQ;YAClB,MAAM,EAAE,yCAAyC;YACjD,OAAO,EAAE,8CAA8C,iBAAiB,YAAY,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE;YAC9G,QAAQ,EAAE;gBACR,aAAa,EAAE,IAAI,CAAC,iBAAiB;gBACrC,aAAa,EAAE,IAAI,CAAC,eAAe;aACpC;SACF,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,WAAW,CAAC,SAAiB,EAAE,MAA2B,EAAE,KAAW;QAC7E,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;QAEhC,MAAM,CAAC,KAAK,CAAC,yBAAyB,EAAE;YACtC,SAAS;YACT,MAAM;YACN,eAAe,EAAE,MAAM,CAAC,eAAe;YACvC,WAAW,EAAE,MAAM,CAAC,WAAW;YAC/B,aAAa,EAAE,IAAI,CAAC,eAAe;YACnC,KAAK,EAAE,KAAK,EAAE,OAAO;SACtB,CAAC,CAAC;QAEH,8CAA8C;QAC9C,IAAI,MAAM,CAAC,eAAe,GAAG,GAAG,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACzD,MAAM,CAAC,IAAI,CAAC,mCAAmC,EAAE;gBAC/C,SAAS;gBACT,eAAe,EAAE,MAAM,CAAC,eAAe;gBACvC,YAAY,EAAE,MAAM,CAAC,YAAY;gBACjC,cAAc,EAAE,uCAAuC;aACxD,CAAC,CAAC;QACL,CAAC;aAAM,IAAI,MAAM,CAAC,eAAe,GAAG,EAAE,IAAI,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC;YAChE,MAAM,CAAC,IAAI,CAAC,qDAAqD,EAAE;gBACjE,SAAS;gBACT,eAAe,EAAE,MAAM,CAAC,eAAe;gBACvC,YAAY,EAAE,MAAM,CAAC,YAAY;gBACjC,cAAc,EAAE,gDAAgD;aACjE,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED;;OAEG;IACH,UAAU;QACR,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC;QAE9C,8BAA8B;QAC9B,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE;YAClC,OAAO,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAC,CAAC;QACrD,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QAEvB,MAAM,CAAC,IAAI,CAAC,kCAAkC,EAAE,EAAE,YAAY,EAAE,CAAC,CAAC;IACpE,CAAC;IAED;;OAEG;IACH,KAAK;QACH,IAAI,CAAC,WAAW,CAAC,KAAK,EAAE,CAAC;QACzB,IAAI,CAAC,UAAU,EAAE,CAAC;QAClB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;QACxB,MAAM,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;IAC3C,CAAC;CACF;AAED,oCAAoC;AACpC,MAAM,CAAC,MAAM,iBAAiB,GAAG,IAAI,iBAAiB,EAAE,CAAC","sourcesContent":["/**\n * GitHubRateLimiter - Specialized rate limiter for GitHub API calls\n * \n * Features:\n * - Respects GitHub's authenticated (5000/hour) and unauthenticated (60/hour) limits\n * - Client-side queuing when approaching limits\n * - Request prioritization for critical operations\n * - Comprehensive logging for quota management\n * - Early termination when exact matches are found\n */\n\nimport { RateLimiter, RateLimiterConfig, RateLimitStatus } from './RateLimiter.js';\nimport { GITHUB_API_RATE_LIMITS } from '../config/portfolio-constants.js';\nimport { TokenManager } from '../security/tokenManager.js';\nimport { logger } from './logger.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\n\nexport interface GitHubRateLimitInfo {\n  limit: number;\n  remaining: number;\n  reset: Date;\n  used: number;\n}\n\nexport interface GitHubApiRequest {\n  id: string;\n  operation: string;\n  priority: 'high' | 'normal' | 'low';\n  timestamp: number;\n  resolve: (value: any) => void;\n  reject: (error: any) => void;\n}\n\nexport interface GitHubRateStatus extends RateLimitStatus {\n  queueLength: number;\n  currentLimit: number;\n  rateLimitInfo?: GitHubRateLimitInfo;\n}\n\nexport class GitHubRateLimiter {\n  private rateLimiter!: RateLimiter;\n  private requestQueue: GitHubApiRequest[] = [];\n  private processing = false;\n  private lastRateLimitInfo?: GitHubRateLimitInfo;\n  private isAuthenticated = false;\n\n  constructor() {\n    // Initialize with conservative limits - will update based on auth status\n    this.updateLimitsForAuthStatus();\n    this.setupPeriodicStatusCheck();\n  }\n\n  /**\n   * Update rate limits based on current authentication status\n   */\n  private async updateLimitsForAuthStatus(): Promise<void> {\n    try {\n      const token = await TokenManager.getGitHubTokenAsync();\n      const newIsAuthenticated = !!token;\n      \n      // Only recreate rate limiter if auth status changed\n      if (newIsAuthenticated !== this.isAuthenticated) {\n        this.isAuthenticated = newIsAuthenticated;\n        \n        const limit = this.isAuthenticated \n          ? GITHUB_API_RATE_LIMITS.AUTHENTICATED_LIMIT \n          : GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT;\n          \n        // Apply buffer to stay below actual limits\n        const bufferedLimit = Math.floor(limit * GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE);\n        \n        const config: RateLimiterConfig = {\n          maxRequests: bufferedLimit,\n          windowMs: GITHUB_API_RATE_LIMITS.WINDOW_MS,\n          minDelayMs: GITHUB_API_RATE_LIMITS.MIN_DELAY_MS\n        };\n        \n        this.rateLimiter = new RateLimiter(config);\n        \n        logger.info('GitHub rate limiter updated', {\n          authenticated: this.isAuthenticated,\n          limit: bufferedLimit,\n          originalLimit: limit,\n          bufferPercentage: GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE\n        });\n      }\n    } catch (error) {\n      logger.warn('Failed to check authentication status for rate limiting', { error });\n      // Fall back to unauthenticated limits\n      this.isAuthenticated = false;\n      this.rateLimiter = new RateLimiter({\n        maxRequests: Math.floor(GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT * GITHUB_API_RATE_LIMITS.BUFFER_PERCENTAGE),\n        windowMs: GITHUB_API_RATE_LIMITS.WINDOW_MS,\n        minDelayMs: GITHUB_API_RATE_LIMITS.MIN_DELAY_MS\n      });\n    }\n  }\n\n  /**\n   * Setup periodic check for rate limit status\n   */\n  private setupPeriodicStatusCheck(): void {\n    // Check auth status every 5 minutes\n    setInterval(() => {\n      this.updateLimitsForAuthStatus().catch(error => {\n        logger.warn('Periodic auth status check failed', { error });\n      });\n    }, 5 * 60 * 1000);\n  }\n\n  /**\n   * Queue a GitHub API request with rate limiting\n   * @param operation Description of the operation\n   * @param apiCall Function that makes the actual API call\n   * @param priority Request priority (high, normal, low)\n   * @returns Promise that resolves with the API response\n   */\n  async queueRequest<T>(\n    operation: string,\n    apiCall: () => Promise<T>,\n    priority: 'high' | 'normal' | 'low' = 'normal'\n  ): Promise<T> {\n    // SECURITY FIX (DMCP-SEC-004): Normalize Unicode in operation name to prevent injection attacks\n    const normalizedOperation = UnicodeValidator.normalize(operation);\n    if (!normalizedOperation.isValid) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'UNICODE_VALIDATION_ERROR',\n        severity: 'MEDIUM',\n        source: 'GitHubRateLimiter.queueRequest',\n        details: `Invalid Unicode in operation name: ${normalizedOperation.detectedIssues?.[0] || 'unknown error'}`\n      });\n      // Use a safe fallback for the operation name\n      operation = 'github-api-request';\n    } else {\n      operation = normalizedOperation.normalizedContent;\n    }\n    \n    const requestId = `${operation}-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n    \n    return new Promise<T>((resolve, reject) => {\n      const request: GitHubApiRequest = {\n        id: requestId,\n        operation,\n        priority,\n        timestamp: Date.now(),\n        resolve: async (value) => {\n          try {\n            logger.debug('Executing GitHub API request', {\n              operation,\n              requestId,\n              queueWaitTime: Date.now() - request.timestamp\n            });\n            \n            const result = await apiCall();\n            resolve(result);\n            \n            // Log successful API usage for quota tracking\n            this.logApiUsage(operation, 'success');\n            \n          } catch (error) {\n            // Check if this is a rate limit error from GitHub\n            if (this.isGitHubRateLimitError(error)) {\n              this.handleGitHubRateLimit(error);\n            }\n            reject(error);\n            this.logApiUsage(operation, 'error', error);\n          }\n        },\n        reject\n      };\n\n      // Add to queue with priority ordering\n      this.addToQueue(request);\n      this.processQueue();\n    });\n  }\n\n  /**\n   * Add request to queue with priority ordering\n   */\n  private addToQueue(request: GitHubApiRequest): void {\n    // Insert based on priority: high > normal > low\n    // Within same priority, maintain FIFO order\n    const priorityOrder = { high: 0, normal: 1, low: 2 };\n    \n    let insertIndex = this.requestQueue.length;\n    for (let i = 0; i < this.requestQueue.length; i++) {\n      if (priorityOrder[request.priority] < priorityOrder[this.requestQueue[i].priority]) {\n        insertIndex = i;\n        break;\n      }\n    }\n    \n    this.requestQueue.splice(insertIndex, 0, request);\n    \n    logger.debug('GitHub API request queued', {\n      operation: request.operation,\n      priority: request.priority,\n      queuePosition: insertIndex,\n      totalQueued: this.requestQueue.length\n    });\n  }\n\n  /**\n   * Process the request queue\n   */\n  private async processQueue(): Promise<void> {\n    if (this.processing || this.requestQueue.length === 0) {\n      return;\n    }\n\n    this.processing = true;\n\n    try {\n      while (this.requestQueue.length > 0) {\n        // Update auth status periodically\n        if (Math.random() < 0.1) { // 10% chance\n          await this.updateLimitsForAuthStatus();\n        }\n\n        const rateLimitStatus = this.rateLimiter.checkLimit();\n        \n        if (!rateLimitStatus.allowed) {\n          // Log rate limit wait\n          logger.info('GitHub API rate limit reached, waiting', {\n            retryAfterMs: rateLimitStatus.retryAfterMs,\n            remainingTokens: rateLimitStatus.remainingTokens,\n            queueLength: this.requestQueue.length,\n            resetTime: rateLimitStatus.resetTime\n          });\n\n          // Wait for the specified time\n          await new Promise(resolve => setTimeout(resolve, rateLimitStatus.retryAfterMs || 1000));\n          continue;\n        }\n\n        // Process the next request\n        const request = this.requestQueue.shift()!;\n        this.rateLimiter.consumeToken();\n\n        // Execute the request\n        request.resolve(null); // This will trigger the actual API call\n      }\n    } finally {\n      this.processing = false;\n    }\n  }\n\n  /**\n   * Get current rate limit status\n   */\n  getStatus(): GitHubRateStatus {\n    const baseStatus = this.rateLimiter.getStatus();\n    \n    return {\n      ...baseStatus,\n      queueLength: this.requestQueue.length,\n      currentLimit: this.isAuthenticated \n        ? GITHUB_API_RATE_LIMITS.AUTHENTICATED_LIMIT \n        : GITHUB_API_RATE_LIMITS.UNAUTHENTICATED_LIMIT,\n      rateLimitInfo: this.lastRateLimitInfo\n    };\n  }\n\n  /**\n   * Check if an error is a GitHub rate limit error\n   */\n  private isGitHubRateLimitError(error: any): boolean {\n    return error?.status === 429 || \n           error?.response?.status === 429 ||\n           (typeof error?.message === 'string' && error.message.toLowerCase().includes('rate limit'));\n  }\n\n  /**\n   * Handle GitHub rate limit error response\n   */\n  private handleGitHubRateLimit(error: any): void {\n    let resetTime: Date | undefined;\n    let remainingRequests = 0;\n    \n    // Parse rate limit headers if available\n    if (error?.response?.headers) {\n      const headers = error.response.headers;\n      const resetTimestamp = parseInt(headers['x-ratelimit-reset'] || '0');\n      const remaining = parseInt(headers['x-ratelimit-remaining'] || '0');\n      const limit = parseInt(headers['x-ratelimit-limit'] || '0');\n      \n      if (resetTimestamp > 0) {\n        resetTime = new Date(resetTimestamp * 1000);\n      }\n      \n      this.lastRateLimitInfo = {\n        limit,\n        remaining,\n        reset: resetTime || new Date(Date.now() + 60 * 60 * 1000), // Default to 1 hour\n        used: limit - remaining\n      };\n      \n      remainingRequests = remaining;\n    }\n\n    logger.warn('GitHub API rate limit hit from server', {\n      remaining: remainingRequests,\n      resetTime,\n      queueLength: this.requestQueue.length,\n      errorMessage: error?.message\n    });\n\n    // Log as a security event for monitoring\n    SecurityMonitor.logSecurityEvent({\n      type: 'RATE_LIMIT_EXCEEDED',\n      severity: 'MEDIUM',\n      source: 'GitHubRateLimiter.handleGitHubRateLimit',\n      details: `GitHub API rate limit exceeded. Remaining: ${remainingRequests}, Queue: ${this.requestQueue.length}`,\n      metadata: {\n        rateLimitInfo: this.lastRateLimitInfo,\n        authenticated: this.isAuthenticated\n      }\n    });\n  }\n\n  /**\n   * Log API usage for monitoring and diagnostics\n   */\n  private logApiUsage(operation: string, result: 'success' | 'error', error?: any): void {\n    const status = this.getStatus();\n    \n    logger.debug('GitHub API usage logged', {\n      operation,\n      result,\n      remainingTokens: status.remainingTokens,\n      queueLength: status.queueLength,\n      authenticated: this.isAuthenticated,\n      error: error?.message\n    });\n\n    // Log warning if getting close to rate limits\n    if (status.remainingTokens < 100 && this.isAuthenticated) {\n      logger.warn('Approaching GitHub API rate limit', {\n        operation,\n        remainingTokens: status.remainingTokens,\n        currentLimit: status.currentLimit,\n        recommendation: 'Consider reducing API usage frequency'\n      });\n    } else if (status.remainingTokens < 10 && !this.isAuthenticated) {\n      logger.warn('Approaching GitHub API rate limit (unauthenticated)', {\n        operation,\n        remainingTokens: status.remainingTokens,\n        currentLimit: status.currentLimit,\n        recommendation: 'Consider authenticating for higher rate limits'\n      });\n    }\n  }\n\n  /**\n   * Clear the request queue (for testing or emergency situations)\n   */\n  clearQueue(): void {\n    const clearedCount = this.requestQueue.length;\n    \n    // Reject all pending requests\n    this.requestQueue.forEach(request => {\n      request.reject(new Error('Request queue cleared'));\n    });\n    \n    this.requestQueue = [];\n    \n    logger.info('GitHub API request queue cleared', { clearedCount });\n  }\n\n  /**\n   * Reset the rate limiter (for testing)\n   */\n  reset(): void {\n    this.rateLimiter.reset();\n    this.clearQueue();\n    this.processing = false;\n    logger.info('GitHub rate limiter reset');\n  }\n}\n\n// Singleton instance for global use\nexport const githubRateLimiter = new GitHubRateLimiter();"]}
|
|
@@ -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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"}
|