@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,201 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Centralized error handling utilities for consistent error processing
|
|
3
|
+
* Preserves stack traces and provides user-friendly messages
|
|
4
|
+
*/
|
|
5
|
+
import { logger } from './logger.js';
|
|
6
|
+
/**
|
|
7
|
+
* Error categories for better error handling
|
|
8
|
+
*/
|
|
9
|
+
export var ErrorCategory;
|
|
10
|
+
(function (ErrorCategory) {
|
|
11
|
+
ErrorCategory["USER_ERROR"] = "USER_ERROR";
|
|
12
|
+
ErrorCategory["SYSTEM_ERROR"] = "SYSTEM_ERROR";
|
|
13
|
+
ErrorCategory["NETWORK_ERROR"] = "NETWORK_ERROR";
|
|
14
|
+
ErrorCategory["AUTH_ERROR"] = "AUTH_ERROR";
|
|
15
|
+
ErrorCategory["VALIDATION_ERROR"] = "VALIDATION_ERROR"; // Validation failures
|
|
16
|
+
})(ErrorCategory || (ErrorCategory = {}));
|
|
17
|
+
/**
|
|
18
|
+
* Custom error class with additional context
|
|
19
|
+
*/
|
|
20
|
+
export class ApplicationError extends Error {
|
|
21
|
+
category;
|
|
22
|
+
code;
|
|
23
|
+
details;
|
|
24
|
+
originalError;
|
|
25
|
+
constructor(message, category = ErrorCategory.SYSTEM_ERROR, code, details, originalError) {
|
|
26
|
+
super(message);
|
|
27
|
+
this.name = 'ApplicationError';
|
|
28
|
+
this.category = category;
|
|
29
|
+
this.code = code;
|
|
30
|
+
this.details = details;
|
|
31
|
+
this.originalError = originalError;
|
|
32
|
+
// Maintain proper stack trace with truncation to prevent memory issues
|
|
33
|
+
if (originalError?.stack) {
|
|
34
|
+
const truncatedOriginalStack = ErrorHandler.truncateStack(originalError.stack);
|
|
35
|
+
this.stack = `${this.stack}\nCaused by: ${truncatedOriginalStack}`;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* Utility class for consistent error handling
|
|
41
|
+
*/
|
|
42
|
+
export class ErrorHandler {
|
|
43
|
+
/**
|
|
44
|
+
* Maximum stack trace depth to prevent memory issues
|
|
45
|
+
*/
|
|
46
|
+
static MAX_STACK_DEPTH = 10;
|
|
47
|
+
/**
|
|
48
|
+
* Maximum length for stack trace strings
|
|
49
|
+
*/
|
|
50
|
+
static MAX_STACK_LENGTH = 5000;
|
|
51
|
+
/**
|
|
52
|
+
* Truncate stack trace to prevent memory issues
|
|
53
|
+
*/
|
|
54
|
+
static truncateStack(stack) {
|
|
55
|
+
if (!stack)
|
|
56
|
+
return undefined;
|
|
57
|
+
// Limit total length
|
|
58
|
+
if (stack.length > this.MAX_STACK_LENGTH) {
|
|
59
|
+
stack = stack.substring(0, this.MAX_STACK_LENGTH) + '\n... (truncated)';
|
|
60
|
+
}
|
|
61
|
+
// Limit number of stack frames
|
|
62
|
+
const lines = stack.split('\n');
|
|
63
|
+
if (lines.length > this.MAX_STACK_DEPTH) {
|
|
64
|
+
return lines.slice(0, this.MAX_STACK_DEPTH).join('\n') + '\n... (truncated)';
|
|
65
|
+
}
|
|
66
|
+
return stack;
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* Extract error information while preserving context
|
|
70
|
+
*/
|
|
71
|
+
static extractErrorInfo(error) {
|
|
72
|
+
// Handle ApplicationError
|
|
73
|
+
if (error instanceof ApplicationError) {
|
|
74
|
+
return {
|
|
75
|
+
message: error.message,
|
|
76
|
+
category: error.category,
|
|
77
|
+
code: error.code,
|
|
78
|
+
details: error.details,
|
|
79
|
+
stack: this.truncateStack(error.stack),
|
|
80
|
+
originalError: error.originalError
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
// Handle standard Error
|
|
84
|
+
if (error instanceof Error) {
|
|
85
|
+
return {
|
|
86
|
+
message: error.message,
|
|
87
|
+
category: ErrorCategory.SYSTEM_ERROR,
|
|
88
|
+
stack: this.truncateStack(error.stack),
|
|
89
|
+
originalError: error
|
|
90
|
+
};
|
|
91
|
+
}
|
|
92
|
+
// Handle string errors
|
|
93
|
+
if (typeof error === 'string') {
|
|
94
|
+
return {
|
|
95
|
+
message: error,
|
|
96
|
+
category: ErrorCategory.SYSTEM_ERROR
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
// Handle unknown errors
|
|
100
|
+
return {
|
|
101
|
+
message: 'An unknown error occurred',
|
|
102
|
+
category: ErrorCategory.SYSTEM_ERROR,
|
|
103
|
+
details: error
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* Get user-friendly error message
|
|
108
|
+
*/
|
|
109
|
+
static getUserMessage(error) {
|
|
110
|
+
const errorInfo = this.extractErrorInfo(error);
|
|
111
|
+
// Provide user-friendly messages based on category
|
|
112
|
+
switch (errorInfo.category) {
|
|
113
|
+
case ErrorCategory.AUTH_ERROR:
|
|
114
|
+
return `Authentication error: ${errorInfo.message}`;
|
|
115
|
+
case ErrorCategory.VALIDATION_ERROR:
|
|
116
|
+
return `Validation error: ${errorInfo.message}`;
|
|
117
|
+
case ErrorCategory.NETWORK_ERROR:
|
|
118
|
+
return `Network error: ${errorInfo.message}. Please check your connection and try again.`;
|
|
119
|
+
case ErrorCategory.USER_ERROR:
|
|
120
|
+
return errorInfo.message; // User errors should already be user-friendly
|
|
121
|
+
default:
|
|
122
|
+
// For system errors, provide a generic message
|
|
123
|
+
return 'An unexpected error occurred. Please try again later.';
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Log error with appropriate level and context
|
|
128
|
+
*/
|
|
129
|
+
static logError(context, error, additionalInfo) {
|
|
130
|
+
const errorInfo = this.extractErrorInfo(error);
|
|
131
|
+
const logData = {
|
|
132
|
+
context,
|
|
133
|
+
category: errorInfo.category,
|
|
134
|
+
code: errorInfo.code,
|
|
135
|
+
message: errorInfo.message,
|
|
136
|
+
...additionalInfo
|
|
137
|
+
};
|
|
138
|
+
// Log based on category
|
|
139
|
+
switch (errorInfo.category) {
|
|
140
|
+
case ErrorCategory.USER_ERROR:
|
|
141
|
+
case ErrorCategory.VALIDATION_ERROR:
|
|
142
|
+
logger.warn(`${context}: ${errorInfo.message}`, logData);
|
|
143
|
+
break;
|
|
144
|
+
case ErrorCategory.AUTH_ERROR:
|
|
145
|
+
logger.warn(`${context}: Authentication error`, logData);
|
|
146
|
+
break;
|
|
147
|
+
case ErrorCategory.NETWORK_ERROR:
|
|
148
|
+
logger.error(`${context}: Network error`, {
|
|
149
|
+
...logData,
|
|
150
|
+
stack: errorInfo.stack
|
|
151
|
+
});
|
|
152
|
+
break;
|
|
153
|
+
default:
|
|
154
|
+
logger.error(`${context}: System error`, {
|
|
155
|
+
...logData,
|
|
156
|
+
stack: errorInfo.stack,
|
|
157
|
+
details: errorInfo.details
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
// Log stack trace in debug mode for all errors (already truncated)
|
|
161
|
+
if (errorInfo.stack) {
|
|
162
|
+
logger.debug(`${context} - Stack trace:`, { stack: errorInfo.stack });
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
* Create an error with context preservation
|
|
167
|
+
*/
|
|
168
|
+
static createError(message, category = ErrorCategory.SYSTEM_ERROR, code, originalError) {
|
|
169
|
+
const original = originalError instanceof Error ? originalError : undefined;
|
|
170
|
+
return new ApplicationError(message, category, code, undefined, original);
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Wrap an error with additional context
|
|
174
|
+
*/
|
|
175
|
+
static wrapError(error, context, category) {
|
|
176
|
+
const errorInfo = this.extractErrorInfo(error);
|
|
177
|
+
return new ApplicationError(`${context}: ${errorInfo.message}`, category || errorInfo.category, errorInfo.code, errorInfo.details, errorInfo.originalError);
|
|
178
|
+
}
|
|
179
|
+
/**
|
|
180
|
+
* Check if error is of a specific category
|
|
181
|
+
*/
|
|
182
|
+
static isErrorCategory(error, category) {
|
|
183
|
+
const errorInfo = this.extractErrorInfo(error);
|
|
184
|
+
return errorInfo.category === category;
|
|
185
|
+
}
|
|
186
|
+
/**
|
|
187
|
+
* Format error for API response
|
|
188
|
+
*/
|
|
189
|
+
static formatForResponse(error) {
|
|
190
|
+
const errorInfo = this.extractErrorInfo(error);
|
|
191
|
+
const userMessage = this.getUserMessage(error);
|
|
192
|
+
return {
|
|
193
|
+
success: false,
|
|
194
|
+
message: userMessage,
|
|
195
|
+
error: errorInfo.code || errorInfo.category,
|
|
196
|
+
// Only include details in development
|
|
197
|
+
details: process.env.NODE_ENV === 'development' ? errorInfo.details : undefined
|
|
198
|
+
};
|
|
199
|
+
}
|
|
200
|
+
}
|
|
201
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ErrorHandler.js","sourceRoot":"","sources":["../../src/utils/ErrorHandler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Centralized error handling utilities for consistent error processing\n * Preserves stack traces and provides user-friendly messages\n */\n\nimport { logger } from './logger.js';\n\n/**\n * Error categories for better error handling\n */\nexport enum ErrorCategory {\n  USER_ERROR = 'USER_ERROR',        // User input issues\n  SYSTEM_ERROR = 'SYSTEM_ERROR',    // Internal system failures\n  NETWORK_ERROR = 'NETWORK_ERROR',  // API/network issues\n  AUTH_ERROR = 'AUTH_ERROR',        // Authentication/authorization\n  VALIDATION_ERROR = 'VALIDATION_ERROR' // Validation failures\n}\n\n/**\n * Structured error information\n */\nexport interface ErrorInfo {\n  message: string;\n  category: ErrorCategory;\n  code?: string;\n  details?: Record<string, unknown>;\n  stack?: string;\n  originalError?: Error;\n}\n\n/**\n * Custom error class with additional context\n */\nexport class ApplicationError extends Error {\n  public readonly category: ErrorCategory;\n  public readonly code?: string;\n  public readonly details?: Record<string, unknown>;\n  public readonly originalError?: Error;\n\n  constructor(\n    message: string,\n    category: ErrorCategory = ErrorCategory.SYSTEM_ERROR,\n    code?: string,\n    details?: Record<string, unknown>,\n    originalError?: Error\n  ) {\n    super(message);\n    this.name = 'ApplicationError';\n    this.category = category;\n    this.code = code;\n    this.details = details;\n    this.originalError = originalError;\n\n    // Maintain proper stack trace with truncation to prevent memory issues\n    if (originalError?.stack) {\n      const truncatedOriginalStack = ErrorHandler.truncateStack(originalError.stack);\n      this.stack = `${this.stack}\\nCaused by: ${truncatedOriginalStack}`;\n    }\n  }\n}\n\n/**\n * Utility class for consistent error handling\n */\nexport class ErrorHandler {\n  /**\n   * Maximum stack trace depth to prevent memory issues\n   */\n  private static readonly MAX_STACK_DEPTH = 10;\n  \n  /**\n   * Maximum length for stack trace strings\n   */\n  private static readonly MAX_STACK_LENGTH = 5000;\n  \n  /**\n   * Truncate stack trace to prevent memory issues\n   */\n  public static truncateStack(stack?: string): string | undefined {\n    if (!stack) return undefined;\n    \n    // Limit total length\n    if (stack.length > this.MAX_STACK_LENGTH) {\n      stack = stack.substring(0, this.MAX_STACK_LENGTH) + '\\n... (truncated)';\n    }\n    \n    // Limit number of stack frames\n    const lines = stack.split('\\n');\n    if (lines.length > this.MAX_STACK_DEPTH) {\n      return lines.slice(0, this.MAX_STACK_DEPTH).join('\\n') + '\\n... (truncated)';\n    }\n    \n    return stack;\n  }\n  \n  /**\n   * Extract error information while preserving context\n   */\n  static extractErrorInfo(error: unknown): ErrorInfo {\n    // Handle ApplicationError\n    if (error instanceof ApplicationError) {\n      return {\n        message: error.message,\n        category: error.category,\n        code: error.code,\n        details: error.details,\n        stack: this.truncateStack(error.stack),\n        originalError: error.originalError\n      };\n    }\n\n    // Handle standard Error\n    if (error instanceof Error) {\n      return {\n        message: error.message,\n        category: ErrorCategory.SYSTEM_ERROR,\n        stack: this.truncateStack(error.stack),\n        originalError: error\n      };\n    }\n\n    // Handle string errors\n    if (typeof error === 'string') {\n      return {\n        message: error,\n        category: ErrorCategory.SYSTEM_ERROR\n      };\n    }\n\n    // Handle unknown errors\n    return {\n      message: 'An unknown error occurred',\n      category: ErrorCategory.SYSTEM_ERROR,\n      details: error as Record<string, unknown> | undefined\n    };\n  }\n\n  /**\n   * Get user-friendly error message\n   */\n  static getUserMessage(error: unknown): string {\n    const errorInfo = this.extractErrorInfo(error);\n    \n    // Provide user-friendly messages based on category\n    switch (errorInfo.category) {\n      case ErrorCategory.AUTH_ERROR:\n        return `Authentication error: ${errorInfo.message}`;\n      case ErrorCategory.VALIDATION_ERROR:\n        return `Validation error: ${errorInfo.message}`;\n      case ErrorCategory.NETWORK_ERROR:\n        return `Network error: ${errorInfo.message}. Please check your connection and try again.`;\n      case ErrorCategory.USER_ERROR:\n        return errorInfo.message; // User errors should already be user-friendly\n      default:\n        // For system errors, provide a generic message\n        return 'An unexpected error occurred. Please try again later.';\n    }\n  }\n\n  /**\n   * Log error with appropriate level and context\n   */\n  static logError(\n    context: string,\n    error: unknown,\n    additionalInfo?: Record<string, unknown>\n  ): void {\n    const errorInfo = this.extractErrorInfo(error);\n    \n    const logData = {\n      context,\n      category: errorInfo.category,\n      code: errorInfo.code,\n      message: errorInfo.message,\n      ...additionalInfo\n    };\n\n    // Log based on category\n    switch (errorInfo.category) {\n      case ErrorCategory.USER_ERROR:\n      case ErrorCategory.VALIDATION_ERROR:\n        logger.warn(`${context}: ${errorInfo.message}`, logData);\n        break;\n      case ErrorCategory.AUTH_ERROR:\n        logger.warn(`${context}: Authentication error`, logData);\n        break;\n      case ErrorCategory.NETWORK_ERROR:\n        logger.error(`${context}: Network error`, {\n          ...logData,\n          stack: errorInfo.stack\n        });\n        break;\n      default:\n        logger.error(`${context}: System error`, {\n          ...logData,\n          stack: errorInfo.stack,\n          details: errorInfo.details\n        });\n    }\n\n    // Log stack trace in debug mode for all errors (already truncated)\n    if (errorInfo.stack) {\n      logger.debug(`${context} - Stack trace:`, { stack: errorInfo.stack });\n    }\n  }\n\n  /**\n   * Create an error with context preservation\n   */\n  static createError(\n    message: string,\n    category: ErrorCategory = ErrorCategory.SYSTEM_ERROR,\n    code?: string,\n    originalError?: unknown\n  ): ApplicationError {\n    const original = originalError instanceof Error ? originalError : undefined;\n    return new ApplicationError(message, category, code, undefined, original);\n  }\n\n  /**\n   * Wrap an error with additional context\n   */\n  static wrapError(\n    error: unknown,\n    context: string,\n    category?: ErrorCategory\n  ): ApplicationError {\n    const errorInfo = this.extractErrorInfo(error);\n    return new ApplicationError(\n      `${context}: ${errorInfo.message}`,\n      category || errorInfo.category,\n      errorInfo.code,\n      errorInfo.details,\n      errorInfo.originalError\n    );\n  }\n\n  /**\n   * Check if error is of a specific category\n   */\n  static isErrorCategory(error: unknown, category: ErrorCategory): boolean {\n    const errorInfo = this.extractErrorInfo(error);\n    return errorInfo.category === category;\n  }\n\n  /**\n   * Format error for API response\n   */\n  static formatForResponse(error: unknown): {\n    success: false;\n    message: string;\n    error: string;\n    details?: Record<string, unknown>;\n  } {\n    const errorInfo = this.extractErrorInfo(error);\n    const userMessage = this.getUserMessage(error);\n    \n    return {\n      success: false,\n      message: userMessage,\n      error: errorInfo.code || errorInfo.category,\n      // Only include details in development\n      details: process.env.NODE_ENV === 'development' ? errorInfo.details : undefined\n    };\n  }\n}"]}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Utility class for efficient file discovery operations
|
|
3
|
+
* Provides optimized file search with caching and security
|
|
4
|
+
*
|
|
5
|
+
* IMPLEMENTATION (PR #503 - PR #496 Recommendation):
|
|
6
|
+
* 1. PERFORMANCE: Single readdir operation instead of multiple file checks
|
|
7
|
+
* 2. PERFORMANCE: 5-second cache TTL for repeated searches
|
|
8
|
+
* 3. SECURITY: Unicode normalization for all search inputs
|
|
9
|
+
* 4. MEMORY: Efficient cache management with 100 entry limit
|
|
10
|
+
*
|
|
11
|
+
* This addresses the PR #496 review recommendation to extract file discovery
|
|
12
|
+
* logic to a reusable utility class for better performance and maintainability.
|
|
13
|
+
*/
|
|
14
|
+
export interface FileSearchOptions {
|
|
15
|
+
extensions?: string[];
|
|
16
|
+
partialMatch?: boolean;
|
|
17
|
+
maxResults?: number;
|
|
18
|
+
cacheResults?: boolean;
|
|
19
|
+
}
|
|
20
|
+
export declare class FileDiscoveryUtil {
|
|
21
|
+
private static readonly CACHE_TTL;
|
|
22
|
+
private static cache;
|
|
23
|
+
/**
|
|
24
|
+
* Find files in a directory with optimized search
|
|
25
|
+
* Uses single readdir operation and filters results
|
|
26
|
+
*/
|
|
27
|
+
static findFiles(directory: string, searchName: string, options?: FileSearchOptions): Promise<string[]>;
|
|
28
|
+
/**
|
|
29
|
+
* Build search patterns for different file extensions
|
|
30
|
+
*/
|
|
31
|
+
private static buildSearchPatterns;
|
|
32
|
+
/**
|
|
33
|
+
* Check if filename matches pattern
|
|
34
|
+
*/
|
|
35
|
+
private static matchesPattern;
|
|
36
|
+
/**
|
|
37
|
+
* Get cached results if still valid
|
|
38
|
+
*/
|
|
39
|
+
private static getCached;
|
|
40
|
+
/**
|
|
41
|
+
* Cache results with timestamp
|
|
42
|
+
*/
|
|
43
|
+
private static setCached;
|
|
44
|
+
/**
|
|
45
|
+
* Clear the cache
|
|
46
|
+
*/
|
|
47
|
+
static clearCache(): void;
|
|
48
|
+
/**
|
|
49
|
+
* Find a single file (convenience method)
|
|
50
|
+
*/
|
|
51
|
+
static findFile(directory: string, searchName: string, options?: FileSearchOptions): Promise<string | null>;
|
|
52
|
+
}
|
|
53
|
+
//# sourceMappingURL=FileDiscoveryUtil.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"FileDiscoveryUtil.d.ts","sourceRoot":"","sources":["../../src/utils/FileDiscoveryUtil.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;GAYG;AAQH,MAAM,WAAW,iBAAiB;IAChC,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;IACtB,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,YAAY,CAAC,EAAE,OAAO,CAAC;CACxB;AAOD,qBAAa,iBAAiB;IAC5B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAQ;IACzC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAiC;IAErD;;;OAGG;WACU,SAAS,CACpB,SAAS,EAAE,MAAM,EACjB,UAAU,EAAE,MAAM,EAClB,OAAO,GAAE,iBAAsB,GAC9B,OAAO,CAAC,MAAM,EAAE,CAAC;IAkFpB;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,mBAAmB;IAoBlC;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,cAAc;IAO7B;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,SAAS;IAaxB;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,SAAS;IAiBxB;;OAEG;IACH,MAAM,CAAC,UAAU,IAAI,IAAI;IAKzB;;OAEG;WACU,QAAQ,CACnB,SAAS,EAAE,MAAM,EACjB,UAAU,EAAE,MAAM,EAClB,OAAO,CAAC,EAAE,iBAAiB,GAC1B,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;CAO1B"}
|
|
@@ -0,0 +1,169 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Utility class for efficient file discovery operations
|
|
3
|
+
* Provides optimized file search with caching and security
|
|
4
|
+
*
|
|
5
|
+
* IMPLEMENTATION (PR #503 - PR #496 Recommendation):
|
|
6
|
+
* 1. PERFORMANCE: Single readdir operation instead of multiple file checks
|
|
7
|
+
* 2. PERFORMANCE: 5-second cache TTL for repeated searches
|
|
8
|
+
* 3. SECURITY: Unicode normalization for all search inputs
|
|
9
|
+
* 4. MEMORY: Efficient cache management with 100 entry limit
|
|
10
|
+
*
|
|
11
|
+
* This addresses the PR #496 review recommendation to extract file discovery
|
|
12
|
+
* logic to a reusable utility class for better performance and maintainability.
|
|
13
|
+
*/
|
|
14
|
+
import * as fs from 'fs/promises';
|
|
15
|
+
import * as path from 'path';
|
|
16
|
+
import { logger } from './logger.js';
|
|
17
|
+
import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
|
|
18
|
+
export class FileDiscoveryUtil {
|
|
19
|
+
static CACHE_TTL = 5000; // 5 seconds cache TTL
|
|
20
|
+
static cache = new Map();
|
|
21
|
+
/**
|
|
22
|
+
* Find files in a directory with optimized search
|
|
23
|
+
* Uses single readdir operation and filters results
|
|
24
|
+
*/
|
|
25
|
+
static async findFiles(directory, searchName, options = {}) {
|
|
26
|
+
const { extensions = ['.json', '.yaml', '.yml', '.md'], partialMatch = true, maxResults = 10, cacheResults = true } = options;
|
|
27
|
+
// Normalize search name for security
|
|
28
|
+
const normalizedSearch = UnicodeValidator.normalize(searchName);
|
|
29
|
+
if (!normalizedSearch.isValid) {
|
|
30
|
+
logger.warn('Invalid Unicode in search name', {
|
|
31
|
+
issues: normalizedSearch.detectedIssues
|
|
32
|
+
});
|
|
33
|
+
return [];
|
|
34
|
+
}
|
|
35
|
+
const safeName = normalizedSearch.normalizedContent.toLowerCase();
|
|
36
|
+
// Check cache if enabled
|
|
37
|
+
const cacheKey = `${directory}:${safeName}:${JSON.stringify(options)}`;
|
|
38
|
+
if (cacheResults) {
|
|
39
|
+
const cached = this.getCached(cacheKey);
|
|
40
|
+
if (cached) {
|
|
41
|
+
logger.debug('File search cache hit', { directory, searchName });
|
|
42
|
+
return cached;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
try {
|
|
46
|
+
// Single readdir operation for efficiency
|
|
47
|
+
const files = await fs.readdir(directory);
|
|
48
|
+
// Build search patterns
|
|
49
|
+
const searchPatterns = this.buildSearchPatterns(safeName, extensions);
|
|
50
|
+
// Filter files efficiently
|
|
51
|
+
const matches = [];
|
|
52
|
+
for (const file of files) {
|
|
53
|
+
const fileLower = file.toLowerCase();
|
|
54
|
+
// Check each pattern
|
|
55
|
+
for (const pattern of searchPatterns) {
|
|
56
|
+
if (this.matchesPattern(fileLower, pattern, partialMatch)) {
|
|
57
|
+
const fullPath = path.join(directory, file);
|
|
58
|
+
matches.push(fullPath);
|
|
59
|
+
if (matches.length >= maxResults) {
|
|
60
|
+
break;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
if (matches.length >= maxResults) {
|
|
65
|
+
break;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
// Cache results if enabled
|
|
69
|
+
if (cacheResults && matches.length > 0) {
|
|
70
|
+
this.setCached(cacheKey, matches);
|
|
71
|
+
}
|
|
72
|
+
// Log file discovery for monitoring
|
|
73
|
+
if (matches.length > 0) {
|
|
74
|
+
logger.debug('Files discovered', {
|
|
75
|
+
directory,
|
|
76
|
+
searchName,
|
|
77
|
+
count: matches.length
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
return matches;
|
|
81
|
+
}
|
|
82
|
+
catch (error) {
|
|
83
|
+
logger.error('File discovery failed', {
|
|
84
|
+
directory,
|
|
85
|
+
searchName,
|
|
86
|
+
error: error instanceof Error ? error.message : String(error)
|
|
87
|
+
});
|
|
88
|
+
return [];
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* Build search patterns for different file extensions
|
|
93
|
+
*/
|
|
94
|
+
static buildSearchPatterns(baseName, extensions) {
|
|
95
|
+
const patterns = [baseName];
|
|
96
|
+
// Add extension variations
|
|
97
|
+
for (const ext of extensions) {
|
|
98
|
+
patterns.push(`${baseName}${ext}`);
|
|
99
|
+
}
|
|
100
|
+
// Add common variations
|
|
101
|
+
const nameWithoutExtension = baseName.replace(/\.[^.]+$/, '');
|
|
102
|
+
if (nameWithoutExtension !== baseName) {
|
|
103
|
+
patterns.push(nameWithoutExtension);
|
|
104
|
+
for (const ext of extensions) {
|
|
105
|
+
patterns.push(`${nameWithoutExtension}${ext}`);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
return patterns;
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Check if filename matches pattern
|
|
112
|
+
*/
|
|
113
|
+
static matchesPattern(filename, pattern, partialMatch) {
|
|
114
|
+
if (partialMatch) {
|
|
115
|
+
return filename.includes(pattern);
|
|
116
|
+
}
|
|
117
|
+
return filename === pattern;
|
|
118
|
+
}
|
|
119
|
+
/**
|
|
120
|
+
* Get cached results if still valid
|
|
121
|
+
*/
|
|
122
|
+
static getCached(key) {
|
|
123
|
+
const entry = this.cache.get(key);
|
|
124
|
+
if (!entry)
|
|
125
|
+
return null;
|
|
126
|
+
const age = Date.now() - entry.timestamp;
|
|
127
|
+
if (age > this.CACHE_TTL) {
|
|
128
|
+
this.cache.delete(key);
|
|
129
|
+
return null;
|
|
130
|
+
}
|
|
131
|
+
return entry.files;
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* Cache results with timestamp
|
|
135
|
+
*/
|
|
136
|
+
static setCached(key, files) {
|
|
137
|
+
// Limit cache size to prevent memory issues
|
|
138
|
+
if (this.cache.size > 100) {
|
|
139
|
+
// Remove oldest entries
|
|
140
|
+
const entries = Array.from(this.cache.entries());
|
|
141
|
+
entries.sort((a, b) => a[1].timestamp - b[1].timestamp);
|
|
142
|
+
for (let i = 0; i < 50; i++) {
|
|
143
|
+
this.cache.delete(entries[i][0]);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
this.cache.set(key, {
|
|
147
|
+
files,
|
|
148
|
+
timestamp: Date.now()
|
|
149
|
+
});
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* Clear the cache
|
|
153
|
+
*/
|
|
154
|
+
static clearCache() {
|
|
155
|
+
this.cache.clear();
|
|
156
|
+
logger.debug('File discovery cache cleared');
|
|
157
|
+
}
|
|
158
|
+
/**
|
|
159
|
+
* Find a single file (convenience method)
|
|
160
|
+
*/
|
|
161
|
+
static async findFile(directory, searchName, options) {
|
|
162
|
+
const files = await this.findFiles(directory, searchName, {
|
|
163
|
+
...options,
|
|
164
|
+
maxResults: 1
|
|
165
|
+
});
|
|
166
|
+
return files.length > 0 ? files[0] : null;
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"FileDiscoveryUtil.js","sourceRoot":"","sources":["../../src/utils/FileDiscoveryUtil.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Utility class for efficient file discovery operations\n * Provides optimized file search with caching and security\n * \n * IMPLEMENTATION (PR #503 - PR #496 Recommendation):\n * 1. PERFORMANCE: Single readdir operation instead of multiple file checks\n * 2. PERFORMANCE: 5-second cache TTL for repeated searches\n * 3. SECURITY: Unicode normalization for all search inputs\n * 4. MEMORY: Efficient cache management with 100 entry limit\n * \n * This addresses the PR #496 review recommendation to extract file discovery\n * logic to a reusable utility class for better performance and maintainability.\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { logger } from './logger.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\n\nexport interface FileSearchOptions {\n  extensions?: string[];\n  partialMatch?: boolean;\n  maxResults?: number;\n  cacheResults?: boolean;\n}\n\ninterface CacheEntry {\n  files: string[];\n  timestamp: number;\n}\n\nexport class FileDiscoveryUtil {\n  private static readonly CACHE_TTL = 5000; // 5 seconds cache TTL\n  private static cache = new Map<string, CacheEntry>();\n  \n  /**\n   * Find files in a directory with optimized search\n   * Uses single readdir operation and filters results\n   */\n  static async findFiles(\n    directory: string,\n    searchName: string,\n    options: FileSearchOptions = {}\n  ): Promise<string[]> {\n    const {\n      extensions = ['.json', '.yaml', '.yml', '.md'],\n      partialMatch = true,\n      maxResults = 10,\n      cacheResults = true\n    } = options;\n    \n    // Normalize search name for security\n    const normalizedSearch = UnicodeValidator.normalize(searchName);\n    if (!normalizedSearch.isValid) {\n      logger.warn('Invalid Unicode in search name', {\n        issues: normalizedSearch.detectedIssues\n      });\n      return [];\n    }\n    const safeName = normalizedSearch.normalizedContent.toLowerCase();\n    \n    // Check cache if enabled\n    const cacheKey = `${directory}:${safeName}:${JSON.stringify(options)}`;\n    if (cacheResults) {\n      const cached = this.getCached(cacheKey);\n      if (cached) {\n        logger.debug('File search cache hit', { directory, searchName });\n        return cached;\n      }\n    }\n    \n    try {\n      // Single readdir operation for efficiency\n      const files = await fs.readdir(directory);\n      \n      // Build search patterns\n      const searchPatterns = this.buildSearchPatterns(safeName, extensions);\n      \n      // Filter files efficiently\n      const matches: string[] = [];\n      for (const file of files) {\n        const fileLower = file.toLowerCase();\n        \n        // Check each pattern\n        for (const pattern of searchPatterns) {\n          if (this.matchesPattern(fileLower, pattern, partialMatch)) {\n            const fullPath = path.join(directory, file);\n            matches.push(fullPath);\n            \n            if (matches.length >= maxResults) {\n              break;\n            }\n          }\n        }\n        \n        if (matches.length >= maxResults) {\n          break;\n        }\n      }\n      \n      // Cache results if enabled\n      if (cacheResults && matches.length > 0) {\n        this.setCached(cacheKey, matches);\n      }\n      \n      // Log file discovery for monitoring\n      if (matches.length > 0) {\n        logger.debug('Files discovered', {\n          directory,\n          searchName,\n          count: matches.length\n        });\n      }\n      \n      return matches;\n    } catch (error) {\n      logger.error('File discovery failed', {\n        directory,\n        searchName,\n        error: error instanceof Error ? error.message : String(error)\n      });\n      return [];\n    }\n  }\n  \n  /**\n   * Build search patterns for different file extensions\n   */\n  private static buildSearchPatterns(baseName: string, extensions: string[]): string[] {\n    const patterns: string[] = [baseName];\n    \n    // Add extension variations\n    for (const ext of extensions) {\n      patterns.push(`${baseName}${ext}`);\n    }\n    \n    // Add common variations\n    const nameWithoutExtension = baseName.replace(/\\.[^.]+$/, '');\n    if (nameWithoutExtension !== baseName) {\n      patterns.push(nameWithoutExtension);\n      for (const ext of extensions) {\n        patterns.push(`${nameWithoutExtension}${ext}`);\n      }\n    }\n    \n    return patterns;\n  }\n  \n  /**\n   * Check if filename matches pattern\n   */\n  private static matchesPattern(filename: string, pattern: string, partialMatch: boolean): boolean {\n    if (partialMatch) {\n      return filename.includes(pattern);\n    }\n    return filename === pattern;\n  }\n  \n  /**\n   * Get cached results if still valid\n   */\n  private static getCached(key: string): string[] | null {\n    const entry = this.cache.get(key);\n    if (!entry) return null;\n    \n    const age = Date.now() - entry.timestamp;\n    if (age > this.CACHE_TTL) {\n      this.cache.delete(key);\n      return null;\n    }\n    \n    return entry.files;\n  }\n  \n  /**\n   * Cache results with timestamp\n   */\n  private static setCached(key: string, files: string[]): void {\n    // Limit cache size to prevent memory issues\n    if (this.cache.size > 100) {\n      // Remove oldest entries\n      const entries = Array.from(this.cache.entries());\n      entries.sort((a, b) => a[1].timestamp - b[1].timestamp);\n      for (let i = 0; i < 50; i++) {\n        this.cache.delete(entries[i][0]);\n      }\n    }\n    \n    this.cache.set(key, {\n      files,\n      timestamp: Date.now()\n    });\n  }\n  \n  /**\n   * Clear the cache\n   */\n  static clearCache(): void {\n    this.cache.clear();\n    logger.debug('File discovery cache cleared');\n  }\n  \n  /**\n   * Find a single file (convenience method)\n   */\n  static async findFile(\n    directory: string,\n    searchName: string,\n    options?: FileSearchOptions\n  ): Promise<string | null> {\n    const files = await this.findFiles(directory, searchName, {\n      ...options,\n      maxResults: 1\n    });\n    return files.length > 0 ? files[0] : null;\n  }\n}"]}
|
|
@@ -0,0 +1,88 @@
|
|
|
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 { RateLimitStatus } from './RateLimiter.js';
|
|
12
|
+
export interface GitHubRateLimitInfo {
|
|
13
|
+
limit: number;
|
|
14
|
+
remaining: number;
|
|
15
|
+
reset: Date;
|
|
16
|
+
used: number;
|
|
17
|
+
}
|
|
18
|
+
export interface GitHubApiRequest {
|
|
19
|
+
id: string;
|
|
20
|
+
operation: string;
|
|
21
|
+
priority: 'high' | 'normal' | 'low';
|
|
22
|
+
timestamp: number;
|
|
23
|
+
resolve: (value: any) => void;
|
|
24
|
+
reject: (error: any) => void;
|
|
25
|
+
}
|
|
26
|
+
export interface GitHubRateStatus extends RateLimitStatus {
|
|
27
|
+
queueLength: number;
|
|
28
|
+
currentLimit: number;
|
|
29
|
+
rateLimitInfo?: GitHubRateLimitInfo;
|
|
30
|
+
}
|
|
31
|
+
export declare class GitHubRateLimiter {
|
|
32
|
+
private rateLimiter;
|
|
33
|
+
private requestQueue;
|
|
34
|
+
private processing;
|
|
35
|
+
private lastRateLimitInfo?;
|
|
36
|
+
private isAuthenticated;
|
|
37
|
+
constructor();
|
|
38
|
+
/**
|
|
39
|
+
* Update rate limits based on current authentication status
|
|
40
|
+
*/
|
|
41
|
+
private updateLimitsForAuthStatus;
|
|
42
|
+
/**
|
|
43
|
+
* Setup periodic check for rate limit status
|
|
44
|
+
*/
|
|
45
|
+
private setupPeriodicStatusCheck;
|
|
46
|
+
/**
|
|
47
|
+
* Queue a GitHub API request with rate limiting
|
|
48
|
+
* @param operation Description of the operation
|
|
49
|
+
* @param apiCall Function that makes the actual API call
|
|
50
|
+
* @param priority Request priority (high, normal, low)
|
|
51
|
+
* @returns Promise that resolves with the API response
|
|
52
|
+
*/
|
|
53
|
+
queueRequest<T>(operation: string, apiCall: () => Promise<T>, priority?: 'high' | 'normal' | 'low'): Promise<T>;
|
|
54
|
+
/**
|
|
55
|
+
* Add request to queue with priority ordering
|
|
56
|
+
*/
|
|
57
|
+
private addToQueue;
|
|
58
|
+
/**
|
|
59
|
+
* Process the request queue
|
|
60
|
+
*/
|
|
61
|
+
private processQueue;
|
|
62
|
+
/**
|
|
63
|
+
* Get current rate limit status
|
|
64
|
+
*/
|
|
65
|
+
getStatus(): GitHubRateStatus;
|
|
66
|
+
/**
|
|
67
|
+
* Check if an error is a GitHub rate limit error
|
|
68
|
+
*/
|
|
69
|
+
private isGitHubRateLimitError;
|
|
70
|
+
/**
|
|
71
|
+
* Handle GitHub rate limit error response
|
|
72
|
+
*/
|
|
73
|
+
private handleGitHubRateLimit;
|
|
74
|
+
/**
|
|
75
|
+
* Log API usage for monitoring and diagnostics
|
|
76
|
+
*/
|
|
77
|
+
private logApiUsage;
|
|
78
|
+
/**
|
|
79
|
+
* Clear the request queue (for testing or emergency situations)
|
|
80
|
+
*/
|
|
81
|
+
clearQueue(): void;
|
|
82
|
+
/**
|
|
83
|
+
* Reset the rate limiter (for testing)
|
|
84
|
+
*/
|
|
85
|
+
reset(): void;
|
|
86
|
+
}
|
|
87
|
+
export declare const githubRateLimiter: GitHubRateLimiter;
|
|
88
|
+
//# sourceMappingURL=GitHubRateLimiter.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"GitHubRateLimiter.d.ts","sourceRoot":"","sources":["../../src/utils/GitHubRateLimiter.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,EAAkC,eAAe,EAAE,MAAM,kBAAkB,CAAC;AAOnF,MAAM,WAAW,mBAAmB;IAClC,KAAK,EAAE,MAAM,CAAC;IACd,SAAS,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,IAAI,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;CACd;AAED,MAAM,WAAW,gBAAgB;IAC/B,EAAE,EAAE,MAAM,CAAC;IACX,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,MAAM,GAAG,QAAQ,GAAG,KAAK,CAAC;IACpC,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,CAAC,KAAK,EAAE,GAAG,KAAK,IAAI,CAAC;IAC9B,MAAM,EAAE,CAAC,KAAK,EAAE,GAAG,KAAK,IAAI,CAAC;CAC9B;AAED,MAAM,WAAW,gBAAiB,SAAQ,eAAe;IACvD,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,MAAM,CAAC;IACrB,aAAa,CAAC,EAAE,mBAAmB,CAAC;CACrC;AAED,qBAAa,iBAAiB;IAC5B,OAAO,CAAC,WAAW,CAAe;IAClC,OAAO,CAAC,YAAY,CAA0B;IAC9C,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,iBAAiB,CAAC,CAAsB;IAChD,OAAO,CAAC,eAAe,CAAS;;IAQhC;;OAEG;YACW,yBAAyB;IA2CvC;;OAEG;IACH,OAAO,CAAC,wBAAwB;IAShC;;;;;;OAMG;IACG,YAAY,CAAC,CAAC,EAClB,SAAS,EAAE,MAAM,EACjB,OAAO,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,EACzB,QAAQ,GAAE,MAAM,GAAG,QAAQ,GAAG,KAAgB,GAC7C,OAAO,CAAC,CAAC,CAAC;IAwDb;;OAEG;IACH,OAAO,CAAC,UAAU;IAuBlB;;OAEG;YACW,YAAY;IA0C1B;;OAEG;IACH,SAAS,IAAI,gBAAgB;IAa7B;;OAEG;IACH,OAAO,CAAC,sBAAsB;IAM9B;;OAEG;IACH,OAAO,CAAC,qBAAqB;IA6C7B;;OAEG;IACH,OAAO,CAAC,WAAW;IA8BnB;;OAEG;IACH,UAAU,IAAI,IAAI;IAalB;;OAEG;IACH,KAAK,IAAI,IAAI;CAMd;AAGD,eAAO,MAAM,iBAAiB,mBAA0B,CAAC"}
|