@dollhousemcp/mcp-server 1.5.2 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/CHANGELOG.md +56 -0
  2. package/README.md +494 -111
  3. package/data/agents/code-reviewer.md +8 -1
  4. package/data/agents/research-assistant.md +8 -1
  5. package/data/agents/task-manager.md +8 -1
  6. package/data/ensembles/business-advisor.md +8 -1
  7. package/data/ensembles/creative-studio.md +8 -1
  8. package/data/ensembles/development-team.md +8 -1
  9. package/data/ensembles/security-analysis-team.md +8 -1
  10. package/data/memories/conversation-history.md +8 -1
  11. package/data/memories/learning-progress.md +8 -1
  12. package/data/memories/project-context.md +8 -1
  13. package/data/personas/business-consultant.md +8 -1
  14. package/data/personas/creative-writer.md +8 -1
  15. package/data/personas/debug-detective.md +8 -1
  16. package/data/personas/eli5-explainer.md +8 -1
  17. package/data/personas/security-analyst.md +8 -1
  18. package/data/personas/technical-analyst.md +8 -1
  19. package/data/skills/code-review.md +8 -1
  20. package/data/skills/creative-writing.md +8 -1
  21. package/data/skills/data-analysis.md +8 -1
  22. package/data/skills/penetration-testing.md +8 -1
  23. package/data/skills/research.md +8 -1
  24. package/data/skills/threat-modeling.md +8 -1
  25. package/data/skills/translation.md +8 -1
  26. package/data/templates/code-documentation.md +8 -1
  27. package/data/templates/email-professional.md +8 -1
  28. package/data/templates/meeting-notes.md +8 -1
  29. package/data/templates/penetration-test-report.md +8 -1
  30. package/data/templates/project-brief.md +8 -1
  31. package/data/templates/report-executive.md +8 -1
  32. package/data/templates/security-vulnerability-report.md +8 -1
  33. package/data/templates/threat-assessment-report.md +8 -1
  34. package/dist/auth/GitHubAuthManager.d.ts +6 -1
  35. package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
  36. package/dist/auth/GitHubAuthManager.js +45 -18
  37. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
  38. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
  39. package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
  40. package/dist/cache/CollectionCache.d.ts.map +1 -1
  41. package/dist/cache/CollectionCache.js +13 -3
  42. package/dist/cache/CollectionIndexCache.d.ts +77 -0
  43. package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
  44. package/dist/cache/CollectionIndexCache.js +349 -0
  45. package/dist/cache/LRUCache.d.ts +93 -0
  46. package/dist/cache/LRUCache.d.ts.map +1 -0
  47. package/dist/cache/LRUCache.js +299 -0
  48. package/dist/cache/index.d.ts +1 -0
  49. package/dist/cache/index.d.ts.map +1 -1
  50. package/dist/cache/index.js +2 -1
  51. package/dist/collection/CollectionBrowser.d.ts +21 -1
  52. package/dist/collection/CollectionBrowser.d.ts.map +1 -1
  53. package/dist/collection/CollectionBrowser.js +130 -10
  54. package/dist/collection/CollectionIndexManager.d.ts +151 -0
  55. package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
  56. package/dist/collection/CollectionIndexManager.js +499 -0
  57. package/dist/collection/CollectionSearch.d.ts +55 -0
  58. package/dist/collection/CollectionSearch.d.ts.map +1 -1
  59. package/dist/collection/CollectionSearch.js +338 -13
  60. package/dist/collection/CollectionSeeder.d.ts.map +1 -1
  61. package/dist/collection/CollectionSeeder.js +38 -1
  62. package/dist/collection/ElementInstaller.d.ts +31 -0
  63. package/dist/collection/ElementInstaller.d.ts.map +1 -1
  64. package/dist/collection/ElementInstaller.js +77 -15
  65. package/dist/collection/PersonaSubmitter.d.ts +1 -1
  66. package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
  67. package/dist/collection/PersonaSubmitter.js +2 -2
  68. package/dist/collection/index.d.ts +1 -0
  69. package/dist/collection/index.d.ts.map +1 -1
  70. package/dist/collection/index.js +2 -1
  71. package/dist/config/ConfigManager.d.ts +78 -0
  72. package/dist/config/ConfigManager.d.ts.map +1 -0
  73. package/dist/config/ConfigManager.js +216 -0
  74. package/dist/config/element-types.d.ts +135 -0
  75. package/dist/config/element-types.d.ts.map +1 -0
  76. package/dist/config/element-types.js +108 -0
  77. package/dist/config/index.d.ts +2 -0
  78. package/dist/config/index.d.ts.map +1 -1
  79. package/dist/config/index.js +3 -1
  80. package/dist/config/portfolio-constants.d.ts +83 -0
  81. package/dist/config/portfolio-constants.d.ts.map +1 -0
  82. package/dist/config/portfolio-constants.js +99 -0
  83. package/dist/elements/BaseElement.d.ts +14 -2
  84. package/dist/elements/BaseElement.d.ts.map +1 -1
  85. package/dist/elements/BaseElement.js +88 -6
  86. package/dist/elements/agents/Agent.d.ts +10 -1
  87. package/dist/elements/agents/Agent.d.ts.map +1 -1
  88. package/dist/elements/agents/Agent.js +66 -19
  89. package/dist/elements/agents/AgentManager.d.ts +2 -0
  90. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  91. package/dist/elements/agents/AgentManager.js +12 -10
  92. package/dist/elements/skills/Skill.d.ts +10 -1
  93. package/dist/elements/skills/Skill.d.ts.map +1 -1
  94. package/dist/elements/skills/Skill.js +40 -3
  95. package/dist/elements/skills/SkillManager.d.ts +1 -0
  96. package/dist/elements/skills/SkillManager.d.ts.map +1 -1
  97. package/dist/elements/skills/SkillManager.js +10 -4
  98. package/dist/elements/templates/Template.d.ts +10 -1
  99. package/dist/elements/templates/Template.d.ts.map +1 -1
  100. package/dist/elements/templates/Template.js +35 -18
  101. package/dist/elements/templates/TemplateManager.d.ts +1 -1
  102. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  103. package/dist/elements/templates/TemplateManager.js +6 -5
  104. package/dist/generated/version.d.ts +2 -2
  105. package/dist/generated/version.js +3 -3
  106. package/dist/index.barrel.d.ts +1 -2
  107. package/dist/index.barrel.d.ts.map +1 -1
  108. package/dist/index.barrel.js +2 -4
  109. package/dist/index.d.ts +143 -25
  110. package/dist/index.d.ts.map +1 -1
  111. package/dist/index.js +1883 -310
  112. package/dist/persona/PersonaElement.d.ts +10 -0
  113. package/dist/persona/PersonaElement.d.ts.map +1 -1
  114. package/dist/persona/PersonaElement.js +55 -32
  115. package/dist/persona/PersonaElementManager.d.ts.map +1 -1
  116. package/dist/persona/PersonaElementManager.js +13 -11
  117. package/dist/persona/PersonaLoader.d.ts.map +1 -1
  118. package/dist/persona/PersonaLoader.js +8 -2
  119. package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
  120. package/dist/persona/export-import/PersonaImporter.js +24 -5
  121. package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
  122. package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
  123. package/dist/persona/export-import/PersonaSharer.js +198 -22
  124. package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
  125. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  126. package/dist/portfolio/DefaultElementProvider.js +499 -7
  127. package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
  128. package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
  129. package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
  130. package/dist/portfolio/MigrationManager.d.ts.map +1 -1
  131. package/dist/portfolio/MigrationManager.js +136 -3
  132. package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
  133. package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
  134. package/dist/portfolio/PortfolioIndexManager.js +478 -0
  135. package/dist/portfolio/PortfolioManager.d.ts +5 -0
  136. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  137. package/dist/portfolio/PortfolioManager.js +61 -20
  138. package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
  139. package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
  140. package/dist/portfolio/PortfolioRepoManager.js +337 -0
  141. package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
  142. package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
  143. package/dist/portfolio/UnifiedIndexManager.js +1434 -0
  144. package/dist/portfolio/index.d.ts +15 -0
  145. package/dist/portfolio/index.d.ts.map +1 -0
  146. package/dist/portfolio/index.js +15 -0
  147. package/dist/portfolio/types.d.ts +7 -0
  148. package/dist/portfolio/types.d.ts.map +1 -1
  149. package/dist/portfolio/types.js +6 -1
  150. package/dist/security/InputValidator.d.ts.map +1 -1
  151. package/dist/security/InputValidator.js +50 -48
  152. package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
  153. package/dist/security/audit/SecurityAuditor.js +17 -9
  154. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  155. package/dist/security/audit/config/suppressions.js +19 -3
  156. package/dist/security/contentValidator.d.ts +2 -0
  157. package/dist/security/contentValidator.d.ts.map +1 -1
  158. package/dist/security/contentValidator.js +115 -4
  159. package/dist/security/secureYamlParser.d.ts +1 -0
  160. package/dist/security/secureYamlParser.d.ts.map +1 -1
  161. package/dist/security/secureYamlParser.js +29 -7
  162. package/dist/security/securityMonitor.d.ts +1 -1
  163. package/dist/security/securityMonitor.d.ts.map +1 -1
  164. package/dist/security/securityMonitor.js +1 -1
  165. package/dist/security/tokenManager.d.ts +1 -1
  166. package/dist/security/tokenManager.d.ts.map +1 -1
  167. package/dist/security/tokenManager.js +30 -10
  168. package/dist/server/ServerSetup.d.ts +22 -2
  169. package/dist/server/ServerSetup.d.ts.map +1 -1
  170. package/dist/server/ServerSetup.js +77 -12
  171. package/dist/server/tools/AuthTools.d.ts.map +1 -1
  172. package/dist/server/tools/AuthTools.js +33 -1
  173. package/dist/server/tools/BuildInfoTools.d.ts +25 -0
  174. package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
  175. package/dist/server/tools/BuildInfoTools.js +36 -0
  176. package/dist/server/tools/CollectionTools.d.ts.map +1 -1
  177. package/dist/server/tools/CollectionTools.js +55 -46
  178. package/dist/server/tools/ConfigTools.d.ts.map +1 -1
  179. package/dist/server/tools/ConfigTools.js +29 -1
  180. package/dist/server/tools/PersonaTools.d.ts +4 -2
  181. package/dist/server/tools/PersonaTools.d.ts.map +1 -1
  182. package/dist/server/tools/PersonaTools.js +5 -152
  183. package/dist/server/tools/PortfolioTools.d.ts +12 -0
  184. package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
  185. package/dist/server/tools/PortfolioTools.js +221 -0
  186. package/dist/server/tools/index.d.ts +3 -1
  187. package/dist/server/tools/index.d.ts.map +1 -1
  188. package/dist/server/tools/index.js +4 -2
  189. package/dist/server/types.d.ts +40 -5
  190. package/dist/server/types.d.ts.map +1 -1
  191. package/dist/server/types.js +1 -1
  192. package/dist/services/BuildInfoService.d.ts +84 -0
  193. package/dist/services/BuildInfoService.d.ts.map +1 -0
  194. package/dist/services/BuildInfoService.js +271 -0
  195. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
  196. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
  197. package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
  198. package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
  199. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
  200. package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
  201. package/dist/tools/portfolio/types.d.ts +41 -0
  202. package/dist/tools/portfolio/types.d.ts.map +1 -0
  203. package/dist/tools/portfolio/types.js +15 -0
  204. package/dist/types/collection.d.ts +51 -0
  205. package/dist/types/collection.d.ts.map +1 -1
  206. package/dist/types/collection.js +1 -1
  207. package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
  208. package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
  209. package/dist/utils/EarlyTerminationSearch.js +164 -0
  210. package/dist/utils/ErrorHandler.d.ts +86 -0
  211. package/dist/utils/ErrorHandler.d.ts.map +1 -0
  212. package/dist/utils/ErrorHandler.js +201 -0
  213. package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
  214. package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
  215. package/dist/utils/FileDiscoveryUtil.js +169 -0
  216. package/dist/utils/GitHubRateLimiter.d.ts +88 -0
  217. package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
  218. package/dist/utils/GitHubRateLimiter.js +315 -0
  219. package/dist/utils/PerformanceMonitor.d.ts +134 -0
  220. package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
  221. package/dist/utils/PerformanceMonitor.js +347 -0
  222. package/dist/utils/RateLimiter.d.ts.map +1 -0
  223. package/dist/utils/RateLimiter.js +172 -0
  224. package/dist/utils/SecureDownloader.d.ts +241 -0
  225. package/dist/utils/SecureDownloader.d.ts.map +1 -0
  226. package/dist/utils/SecureDownloader.js +759 -0
  227. package/dist/utils/ToolCache.d.ts +82 -0
  228. package/dist/utils/ToolCache.d.ts.map +1 -0
  229. package/dist/utils/ToolCache.js +196 -0
  230. package/dist/utils/errorCodes.d.ts +136 -0
  231. package/dist/utils/errorCodes.d.ts.map +1 -0
  232. package/dist/utils/errorCodes.js +87 -0
  233. package/dist/utils/index.d.ts +3 -0
  234. package/dist/utils/index.d.ts.map +1 -1
  235. package/dist/utils/index.js +4 -1
  236. package/dist/utils/installation.d.ts +1 -1
  237. package/dist/utils/installation.d.ts.map +1 -1
  238. package/dist/utils/installation.js +9 -8
  239. package/dist/utils/searchUtils.d.ts +31 -0
  240. package/dist/utils/searchUtils.d.ts.map +1 -1
  241. package/dist/utils/searchUtils.js +62 -1
  242. package/package.json +17 -7
  243. package/dist/config/updateConfig.d.ts +0 -84
  244. package/dist/config/updateConfig.d.ts.map +0 -1
  245. package/dist/config/updateConfig.js +0 -148
  246. package/dist/server/tools/UpdateTools.d.ts +0 -10
  247. package/dist/server/tools/UpdateTools.d.ts.map +0 -1
  248. package/dist/server/tools/UpdateTools.js +0 -85
  249. package/dist/update/BackupManager.d.ts +0 -63
  250. package/dist/update/BackupManager.d.ts.map +0 -1
  251. package/dist/update/BackupManager.js +0 -370
  252. package/dist/update/DependencyChecker.d.ts +0 -41
  253. package/dist/update/DependencyChecker.d.ts.map +0 -1
  254. package/dist/update/DependencyChecker.js +0 -132
  255. package/dist/update/RateLimiter.d.ts.map +0 -1
  256. package/dist/update/RateLimiter.js +0 -172
  257. package/dist/update/SignatureVerifier.d.ts +0 -71
  258. package/dist/update/SignatureVerifier.d.ts.map +0 -1
  259. package/dist/update/SignatureVerifier.js +0 -214
  260. package/dist/update/UpdateChecker.d.ts +0 -132
  261. package/dist/update/UpdateChecker.d.ts.map +0 -1
  262. package/dist/update/UpdateChecker.js +0 -506
  263. package/dist/update/UpdateManager.d.ts +0 -60
  264. package/dist/update/UpdateManager.d.ts.map +0 -1
  265. package/dist/update/UpdateManager.js +0 -730
  266. package/dist/update/VersionManager.d.ts +0 -31
  267. package/dist/update/VersionManager.d.ts.map +0 -1
  268. package/dist/update/VersionManager.js +0 -181
  269. package/dist/update/index.d.ts +0 -9
  270. package/dist/update/index.d.ts.map +0 -1
  271. package/dist/update/index.js +0 -9
  272. /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
@@ -5,6 +5,8 @@ import * as path from 'path';
5
5
  import { SECURITY_LIMITS, VALIDATION_PATTERNS } from './constants.js';
6
6
  import { VALID_CATEGORIES } from '../config/constants.js';
7
7
  import { RegexValidator } from './regexValidator.js';
8
+ import { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';
9
+ import { ValidationErrorCodes } from '../utils/errorCodes.js';
8
10
  // Pre-compiled regex patterns for better performance
9
11
  // These patterns are used repeatedly and benefit from pre-compilation
10
12
  const CONTROL_CHARS_REGEX = /[\x00-\x1F\x7F]/g;
@@ -31,15 +33,15 @@ export class MCPInputValidator {
31
33
  */
32
34
  static validatePersonaIdentifier(identifier) {
33
35
  if (!identifier || typeof identifier !== 'string') {
34
- throw new Error('Persona identifier must be a non-empty string');
36
+ throw ErrorHandler.createError('Persona identifier must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PERSONA_ID);
35
37
  }
36
38
  if (identifier.length > 100) {
37
- throw new Error('Persona identifier too long (max 100 characters)');
39
+ throw ErrorHandler.createError('Persona identifier too long (max 100 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);
38
40
  }
39
41
  // Allow persona names and filenames
40
42
  const sanitized = sanitizeInput(identifier, 100);
41
43
  if (!sanitized) {
42
- throw new Error('Persona identifier contains only invalid characters');
44
+ throw ErrorHandler.createError('Persona identifier contains only invalid characters', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PERSONA_ID);
43
45
  }
44
46
  return sanitized;
45
47
  }
@@ -48,13 +50,13 @@ export class MCPInputValidator {
48
50
  */
49
51
  static validateSearchQuery(query) {
50
52
  if (!query || typeof query !== 'string') {
51
- throw new Error('Search query must be a non-empty string');
53
+ throw ErrorHandler.createError('Search query must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_SEARCH_QUERY);
52
54
  }
53
55
  if (query.length < 2) {
54
- throw new Error('Search query too short (minimum 2 characters)');
56
+ throw ErrorHandler.createError('Search query too short (minimum 2 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);
55
57
  }
56
58
  if (query.length > 200) {
57
- throw new Error('Search query too long (max 200 characters)');
59
+ throw ErrorHandler.createError('Search query too long (max 200 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);
58
60
  }
59
61
  // Sanitize but preserve spaces for search
60
62
  const sanitized = query
@@ -64,7 +66,7 @@ export class MCPInputValidator {
64
66
  .replace(RTL_ZEROWIDTH_REGEX, '') // Remove RTL override and zero-width chars
65
67
  .trim();
66
68
  if (!sanitized) {
67
- throw new Error('Search query contains only invalid characters');
69
+ throw ErrorHandler.createError('Search query contains only invalid characters', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_SEARCH_QUERY);
68
70
  }
69
71
  return sanitized;
70
72
  }
@@ -73,10 +75,10 @@ export class MCPInputValidator {
73
75
  */
74
76
  static validateCollectionPath(path) {
75
77
  if (!path || typeof path !== 'string') {
76
- throw new Error('Collection path must be a non-empty string');
78
+ throw ErrorHandler.createError('Collection path must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_COLLECTION_PATH);
77
79
  }
78
80
  if (path.length > 500) {
79
- throw new Error('Collection path too long (max 500 characters)');
81
+ throw ErrorHandler.createError('Collection path too long (max 500 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);
80
82
  }
81
83
  // GitHub API paths should be safe filename patterns
82
84
  // Use single regex test for better performance (avoids O(n) character-by-character check)
@@ -85,11 +87,11 @@ export class MCPInputValidator {
85
87
  for (let i = 0; i < path.length; i++) {
86
88
  const char = path[i];
87
89
  if (!COLLECTION_PATH_CHAR_REGEX.test(char)) {
88
- throw new Error(`Invalid character '${char}' in collection path at position ${i + 1}`);
90
+ throw ErrorHandler.createError(`Invalid character '${char}' in collection path at position ${i + 1}`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_CHARACTER);
89
91
  }
90
92
  }
91
93
  // Fallback error if we somehow don't find the invalid character
92
- throw new Error('Invalid characters in collection path');
94
+ throw ErrorHandler.createError('Invalid characters in collection path', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_COLLECTION_PATH);
93
95
  }
94
96
  // Prevent path traversal in GitHub paths (comprehensive check)
95
97
  const pathLower = path.toLowerCase();
@@ -111,7 +113,7 @@ export class MCPInputValidator {
111
113
  ];
112
114
  for (const pattern of traversalPatterns) {
113
115
  if (pathLower.includes(pattern) || encodedPath.toLowerCase().includes(pattern)) {
114
- throw new Error('Path traversal not allowed in collection path');
116
+ throw ErrorHandler.createError('Path traversal not allowed in collection path', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.PATH_TRAVERSAL);
115
117
  }
116
118
  }
117
119
  return path;
@@ -121,14 +123,14 @@ export class MCPInputValidator {
121
123
  */
122
124
  static validateImportUrl(url) {
123
125
  if (!url || typeof url !== 'string') {
124
- throw new Error('URL must be a non-empty string');
126
+ throw ErrorHandler.createError('URL must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);
125
127
  }
126
128
  if (url.length > 2000) {
127
- throw new Error('URL too long (max 2000 characters)');
129
+ throw ErrorHandler.createError('URL too long (max 2000 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);
128
130
  }
129
131
  // Reject protocol-relative URLs that could bypass validation
130
132
  if (url.startsWith('//')) {
131
- throw new Error('Protocol-relative URLs are not allowed');
133
+ throw ErrorHandler.createError('Protocol-relative URLs are not allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);
132
134
  }
133
135
  try {
134
136
  // Decode URL to prevent encoding-based bypasses
@@ -142,7 +144,7 @@ export class MCPInputValidator {
142
144
  const parsed = new URL(decodedUrl);
143
145
  // Protocol validation
144
146
  if (!['http:', 'https:'].includes(parsed.protocol)) {
145
- throw new Error('Only HTTP(S) URLs are allowed');
147
+ throw ErrorHandler.createError('Only HTTP(S) URLs are allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);
146
148
  }
147
149
  // Enhanced SSRF protection with IDN normalization
148
150
  let hostname = parsed.hostname.toLowerCase();
@@ -153,15 +155,15 @@ export class MCPInputValidator {
153
155
  }
154
156
  catch (idnError) {
155
157
  // If IDN conversion fails, reject the URL for security
156
- throw new Error('Invalid hostname: IDN conversion failed - potentially malicious domain name');
158
+ throw ErrorHandler.createError('Invalid hostname: IDN conversion failed - potentially malicious domain name', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);
157
159
  }
158
160
  // Check for private IPs (now with IDN-normalized hostname)
159
161
  if (this.isPrivateIP(hostname)) {
160
- throw new Error('Private network URLs are not allowed');
162
+ throw ErrorHandler.createError('Private network URLs are not allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);
161
163
  }
162
164
  // Additional SSRF checks for encoded IPs
163
165
  if (this.isEncodedPrivateIP(hostname)) {
164
- throw new Error('Encoded private network URLs are not allowed');
166
+ throw ErrorHandler.createError('Encoded private network URLs are not allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);
165
167
  }
166
168
  return url;
167
169
  }
@@ -170,7 +172,7 @@ export class MCPInputValidator {
170
172
  throw error;
171
173
  }
172
174
  const errorMessage = error instanceof Error ? error.message : 'Unknown error';
173
- throw new Error(`Invalid URL format: ${errorMessage}`);
175
+ throw ErrorHandler.createError(`Invalid URL format: ${errorMessage}`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);
174
176
  }
175
177
  }
176
178
  /**
@@ -178,13 +180,13 @@ export class MCPInputValidator {
178
180
  */
179
181
  static validateExpiryDays(days) {
180
182
  if (typeof days !== 'number') {
181
- throw new Error('Expiry days must be a valid number');
183
+ throw ErrorHandler.createError('Expiry days must be a valid number', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_NUMBER);
182
184
  }
183
185
  if (isNaN(days) || !isFinite(days)) {
184
- throw new Error('Expiry days must be a valid number');
186
+ throw ErrorHandler.createError('Expiry days must be a valid number', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_NUMBER);
185
187
  }
186
188
  if (days < 1 || days > 365) {
187
- throw new Error('Expiry days must be between 1 and 365');
189
+ throw ErrorHandler.createError('Expiry days must be between 1 and 365', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_RANGE);
188
190
  }
189
191
  return Math.floor(days);
190
192
  }
@@ -193,10 +195,10 @@ export class MCPInputValidator {
193
195
  */
194
196
  static validateConfirmation(confirm, operationName) {
195
197
  if (typeof confirm !== 'boolean') {
196
- throw new Error(`${operationName} confirmation must be a boolean value`);
198
+ throw ErrorHandler.createError(`${operationName} confirmation must be a boolean value`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_TYPE);
197
199
  }
198
200
  if (!confirm) {
199
- throw new Error(`${operationName} operation requires explicit confirmation (true)`);
201
+ throw ErrorHandler.createError(`${operationName} operation requires explicit confirmation (true)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.CONFIRMATION_REQUIRED);
200
202
  }
201
203
  return confirm;
202
204
  }
@@ -205,7 +207,7 @@ export class MCPInputValidator {
205
207
  */
206
208
  static validateEditField(field) {
207
209
  if (!field || typeof field !== 'string') {
208
- throw new Error('Field name must be a non-empty string');
210
+ throw ErrorHandler.createError('Field name must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.REQUIRED_FIELD);
209
211
  }
210
212
  const validFields = [
211
213
  'name', 'description', 'category', 'instructions',
@@ -213,7 +215,7 @@ export class MCPInputValidator {
213
215
  ];
214
216
  const normalizedField = field.toLowerCase().trim();
215
217
  if (!validFields.includes(normalizedField)) {
216
- throw new Error(`Invalid field name. Must be one of: ${validFields.join(', ')}`);
218
+ throw ErrorHandler.createError(`Invalid field name. Must be one of: ${validFields.join(', ')}`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_INPUT);
217
219
  }
218
220
  return normalizedField;
219
221
  }
@@ -297,15 +299,15 @@ export class MCPInputValidator {
297
299
  */
298
300
  export function validateFilename(filename) {
299
301
  if (!filename || typeof filename !== 'string') {
300
- throw new Error('Filename must be a non-empty string');
302
+ throw ErrorHandler.createError('Filename must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_FILENAME);
301
303
  }
302
304
  if (filename.length > SECURITY_LIMITS.MAX_FILENAME_LENGTH) {
303
- throw new Error(`Filename too long (max ${SECURITY_LIMITS.MAX_FILENAME_LENGTH} characters)`);
305
+ throw ErrorHandler.createError(`Filename too long (max ${SECURITY_LIMITS.MAX_FILENAME_LENGTH} characters)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);
304
306
  }
305
307
  // Remove any path separators and dangerous characters
306
308
  const sanitized = filename.replace(FILENAME_DANGEROUS_REGEX, '').replace(FILENAME_LEADING_DOTS_REGEX, '');
307
309
  if (!RegexValidator.validate(sanitized, VALIDATION_PATTERNS.SAFE_FILENAME, { maxLength: SECURITY_LIMITS.MAX_FILENAME_LENGTH })) {
308
- throw new Error('Invalid filename format. Use alphanumeric characters, hyphens, underscores, and dots only.');
310
+ throw ErrorHandler.createError('Invalid filename format. Use alphanumeric characters, hyphens, underscores, and dots only.', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_FILENAME);
309
311
  }
310
312
  return sanitized;
311
313
  }
@@ -314,14 +316,14 @@ export function validateFilename(filename) {
314
316
  */
315
317
  export function validatePath(inputPath, baseDir) {
316
318
  if (!inputPath || typeof inputPath !== 'string') {
317
- throw new Error('Path must be a non-empty string');
319
+ throw ErrorHandler.createError('Path must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PATH);
318
320
  }
319
321
  // If baseDir is provided and inputPath is absolute, reject it
320
322
  // Check both Unix-style and Windows-style absolute paths for cross-platform security
321
323
  const isUnixAbsolute = path.isAbsolute(inputPath);
322
324
  const isWindowsAbsolute = /^[a-zA-Z]:[\\/]/.test(inputPath);
323
325
  if (baseDir && (isUnixAbsolute || isWindowsAbsolute)) {
324
- throw new Error('Absolute paths not allowed when base directory is specified');
326
+ throw ErrorHandler.createError('Absolute paths not allowed when base directory is specified', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PATH);
325
327
  }
326
328
  // Remove leading/trailing slashes and normalize
327
329
  // Length limits added to prevent ReDoS attacks
@@ -336,23 +338,23 @@ export function validatePath(inputPath, baseDir) {
336
338
  normalized = '/' + normalized;
337
339
  }
338
340
  if (!VALIDATION_PATTERNS.SAFE_PATH.test(normalized)) {
339
- throw new Error('Invalid path format. Use alphanumeric characters, hyphens, underscores, dots, and forward slashes only.');
341
+ throw ErrorHandler.createError('Invalid path format. Use alphanumeric characters, hyphens, underscores, dots, and forward slashes only.', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PATH);
340
342
  }
341
343
  // Check for path traversal attempts
342
344
  if (normalized.includes('..') || normalized.includes('./') || normalized.includes('/.')) {
343
- throw new Error('Path traversal not allowed');
345
+ throw ErrorHandler.createError('Path traversal not allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.PATH_TRAVERSAL);
344
346
  }
345
347
  // Validate path depth
346
348
  const depth = normalized.split('/').length;
347
349
  if (depth > SECURITY_LIMITS.MAX_PATH_DEPTH) {
348
- throw new Error(`Path too deep (max ${SECURITY_LIMITS.MAX_PATH_DEPTH} levels)`);
350
+ throw ErrorHandler.createError(`Path too deep (max ${SECURITY_LIMITS.MAX_PATH_DEPTH} levels)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PATH);
349
351
  }
350
352
  // If baseDir provided, ensure path is within it
351
353
  if (baseDir) {
352
354
  const resolvedPath = path.resolve(baseDir, normalized);
353
355
  const resolvedBase = path.resolve(baseDir);
354
356
  if (!resolvedPath.startsWith(resolvedBase)) {
355
- throw new Error('Path traversal attempt detected');
357
+ throw ErrorHandler.createError('Path traversal attempt detected', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.PATH_TRAVERSAL);
356
358
  }
357
359
  }
358
360
  return normalized;
@@ -362,10 +364,10 @@ export function validatePath(inputPath, baseDir) {
362
364
  */
363
365
  export function validateUsername(username) {
364
366
  if (!username || typeof username !== 'string') {
365
- throw new Error('Username must be a non-empty string');
367
+ throw ErrorHandler.createError('Username must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.REQUIRED_FIELD);
366
368
  }
367
369
  if (!VALIDATION_PATTERNS.SAFE_USERNAME.test(username)) {
368
- throw new Error('Invalid username format. Use alphanumeric characters, hyphens, underscores, and dots only.');
370
+ throw ErrorHandler.createError('Invalid username format. Use alphanumeric characters, hyphens, underscores, and dots only.', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_FORMAT);
369
371
  }
370
372
  return username.toLowerCase();
371
373
  }
@@ -374,14 +376,14 @@ export function validateUsername(username) {
374
376
  */
375
377
  export function validateCategory(category) {
376
378
  if (!category || typeof category !== 'string') {
377
- throw new Error('Category must be a non-empty string');
379
+ throw ErrorHandler.createError('Category must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_CATEGORY);
378
380
  }
379
381
  if (!RegexValidator.validate(category, VALIDATION_PATTERNS.SAFE_CATEGORY, { maxLength: 50 })) {
380
- throw new Error('Invalid category format. Use alphabetic characters, hyphens, and underscores only.');
382
+ throw ErrorHandler.createError('Invalid category format. Use alphabetic characters, hyphens, and underscores only.', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_CATEGORY);
381
383
  }
382
384
  const normalized = category.toLowerCase();
383
385
  if (!VALID_CATEGORIES.includes(normalized)) {
384
- throw new Error(`Invalid category. Must be one of: ${VALID_CATEGORIES.join(', ')}`);
386
+ throw ErrorHandler.createError(`Invalid category. Must be one of: ${VALID_CATEGORIES.join(', ')}`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_CATEGORY);
385
387
  }
386
388
  return normalized;
387
389
  }
@@ -390,11 +392,11 @@ export function validateCategory(category) {
390
392
  */
391
393
  export function validateContentSize(content, maxSize = SECURITY_LIMITS.MAX_CONTENT_LENGTH) {
392
394
  if (!content || typeof content !== 'string') {
393
- throw new Error('Content must be a non-empty string');
395
+ throw ErrorHandler.createError('Content must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.REQUIRED_FIELD);
394
396
  }
395
397
  const sizeBytes = Buffer.byteLength(content, 'utf8');
396
398
  if (sizeBytes > maxSize) {
397
- throw new Error(`Content too large (${sizeBytes} bytes, max ${maxSize} bytes)`);
399
+ throw ErrorHandler.createError(`Content too large (${sizeBytes} bytes, max ${maxSize} bytes)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.CONTENT_TOO_LARGE);
398
400
  }
399
401
  }
400
402
  export function validateInputLengths(content, contentType, options = {}) {
@@ -408,23 +410,23 @@ export function validateInputLengths(content, contentType, options = {}) {
408
410
  switch (contentType) {
409
411
  case 'full':
410
412
  if (content.length > limits.maxContentLength) {
411
- throw new Error(`Content exceeds maximum length of ${limits.maxContentLength} characters (${content.length} provided)`);
413
+ throw ErrorHandler.createError(`Content exceeds maximum length of ${limits.maxContentLength} characters (${content.length} provided)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.CONTENT_TOO_LARGE);
412
414
  }
413
415
  break;
414
416
  case 'yaml':
415
417
  if (content.length > limits.maxYamlLength) {
416
- throw new Error(`YAML content exceeds maximum length of ${limits.maxYamlLength} characters (${content.length} provided)`);
418
+ throw ErrorHandler.createError(`YAML content exceeds maximum length of ${limits.maxYamlLength} characters (${content.length} provided)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.CONTENT_TOO_LARGE);
417
419
  }
418
420
  break;
419
421
  case 'metadata':
420
422
  // For metadata, check overall size
421
423
  if (content.length > limits.maxYamlLength) {
422
- throw new Error(`Metadata exceeds maximum length of ${limits.maxYamlLength} characters (${content.length} provided)`);
424
+ throw ErrorHandler.createError(`Metadata exceeds maximum length of ${limits.maxYamlLength} characters (${content.length} provided)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.CONTENT_TOO_LARGE);
423
425
  }
424
426
  break;
425
427
  case 'field':
426
428
  if (content.length > limits.maxMetadataFieldLength) {
427
- throw new Error(`Field exceeds maximum length of ${limits.maxMetadataFieldLength} characters (${content.length} provided)`);
429
+ throw ErrorHandler.createError(`Field exceeds maximum length of ${limits.maxMetadataFieldLength} characters (${content.length} provided)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.CONTENT_TOO_LARGE);
428
430
  }
429
431
  break;
430
432
  }
@@ -445,4 +447,4 @@ export function sanitizeInput(input, maxLength = 1000) {
445
447
  .substring(0, maxLength)
446
448
  .trim();
447
449
  }
448
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"InputValidator.js","sourceRoot":"","sources":["../../src/security/InputValidator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Input validation and sanitization functions\n */\n\nimport * as path from 'path';\nimport { SECURITY_LIMITS, VALIDATION_PATTERNS } from './constants.js';\nimport { VALID_CATEGORIES } from '../config/constants.js';\nimport { RegexValidator } from './regexValidator.js';\n\n// Pre-compiled regex patterns for better performance\n// These patterns are used repeatedly and benefit from pre-compilation\nconst CONTROL_CHARS_REGEX = /[\\x00-\\x1F\\x7F]/g;\nconst HTML_DANGEROUS_REGEX = /[<>'\"&]/g;\nconst SHELL_METACHAR_REGEX = /[;&|`$()!\\\\~*?{}]/g;\nconst RTL_ZEROWIDTH_REGEX = /[\\u202E\\uFEFF]/g;\nconst COLLECTION_PATH_CHAR_REGEX = /[a-zA-Z0-9\\/\\-_.]/;\nconst VALID_COLLECTION_PATH_REGEX = /^[a-zA-Z0-9\\/\\-_.]*$/;\nconst IPV4_REGEX = /^(\\d+)\\.(\\d+)\\.(\\d+)\\.(\\d+)$/;\nconst DECIMAL_IP_REGEX = /^\\d{8,10}$/;\nconst HEX_IP_REGEX = /^0x[0-9a-f]{1,8}$/i;\nconst OCTAL_IP_REGEX = /^0[0-7]{8,11}$/;\nconst FILENAME_DANGEROUS_REGEX = /[\\/\\\\:*?\"<>|]/g;\nconst FILENAME_LEADING_DOTS_REGEX = /^\\.+/;\nconst PATH_NORMALIZE_REGEX = /^\\/{1,100}|\\/{1,100}$/g;\nconst PATH_MULTIPLE_SLASHES_REGEX = /\\/{1,100}/g;\nconst URL_PLUS_DECODE_REGEX = /\\+/g;\n\n/**\n * Enhanced input validation for MCP tools\n */\nexport class MCPInputValidator {\n  /**\n   * Validate a persona identifier (name or filename)\n   */\n  static validatePersonaIdentifier(identifier: string): string {\n    if (!identifier || typeof identifier !== 'string') {\n      throw new Error('Persona identifier must be a non-empty string');\n    }\n\n    if (identifier.length > 100) {\n      throw new Error('Persona identifier too long (max 100 characters)');\n    }\n\n    // Allow persona names and filenames\n    const sanitized = sanitizeInput(identifier, 100);\n    if (!sanitized) {\n      throw new Error('Persona identifier contains only invalid characters');\n    }\n\n    return sanitized;\n  }\n\n  /**\n   * Validate search query for collection\n   */\n  static validateSearchQuery(query: string): string {\n    if (!query || typeof query !== 'string') {\n      throw new Error('Search query must be a non-empty string');\n    }\n\n    if (query.length < 2) {\n      throw new Error('Search query too short (minimum 2 characters)');\n    }\n\n    if (query.length > 200) {\n      throw new Error('Search query too long (max 200 characters)');\n    }\n\n    // Sanitize but preserve spaces for search\n    const sanitized = query\n      .replace(CONTROL_CHARS_REGEX, '') // Remove control characters\n      .replace(HTML_DANGEROUS_REGEX, '') // Remove HTML-dangerous characters\n      .replace(SHELL_METACHAR_REGEX, '') // Remove shell metacharacters (expanded)\n      .replace(RTL_ZEROWIDTH_REGEX, '') // Remove RTL override and zero-width chars\n      .trim();\n\n    if (!sanitized) {\n      throw new Error('Search query contains only invalid characters');\n    }\n\n    return sanitized;\n  }\n\n  /**\n   * Validate collection path\n   */\n  static validateCollectionPath(path: string): string {\n    if (!path || typeof path !== 'string') {\n      throw new Error('Collection path must be a non-empty string');\n    }\n\n    if (path.length > 500) {\n      throw new Error('Collection path too long (max 500 characters)');\n    }\n\n    // GitHub API paths should be safe filename patterns\n    // Use single regex test for better performance (avoids O(n) character-by-character check)\n    if (!VALID_COLLECTION_PATH_REGEX.test(path)) {\n      // Only do character-by-character check if validation fails, to provide detailed error message\n      for (let i = 0; i < path.length; i++) {\n        const char = path[i];\n        if (!COLLECTION_PATH_CHAR_REGEX.test(char)) {\n          throw new Error(`Invalid character '${char}' in collection path at position ${i + 1}`);\n        }\n      }\n      // Fallback error if we somehow don't find the invalid character\n      throw new Error('Invalid characters in collection path');\n    }\n\n    // Prevent path traversal in GitHub paths (comprehensive check)\n    const pathLower = path.toLowerCase();\n    const encodedPath = decodeURIComponent(path.replace(URL_PLUS_DECODE_REGEX, ' ')); // Decode URL encoding\n    \n    // Check for various path traversal patterns\n    const traversalPatterns = [\n      '..',          // Basic traversal\n      './',          // Current directory\n      '/../',        // Directory traversal with slashes\n      '\\\\',          // Backslash (Windows-style)\n      '%2e%2e',      // URL-encoded ..\n      '%2e%2e%2f',   // URL-encoded ../\n      '%2e%2e%5c',   // URL-encoded ..\\\n      '%252e%252e',  // Double URL-encoded ..\n      '..%2f',       // Mixed encoding\n      '..%5c',       // Mixed encoding with backslash\n      '..../',       // Dotdot bypass attempt\n      '..;/',        // Semicolon bypass attempt\n    ];\n    \n    for (const pattern of traversalPatterns) {\n      if (pathLower.includes(pattern) || encodedPath.toLowerCase().includes(pattern)) {\n        throw new Error('Path traversal not allowed in collection path');\n      }\n    }\n\n    return path;\n  }\n\n  /**\n   * Validate URL for import operations\n   */\n  static validateImportUrl(url: string): string {\n    if (!url || typeof url !== 'string') {\n      throw new Error('URL must be a non-empty string');\n    }\n\n    if (url.length > 2000) {\n      throw new Error('URL too long (max 2000 characters)');\n    }\n\n    // Reject protocol-relative URLs that could bypass validation\n    if (url.startsWith('//')) {\n      throw new Error('Protocol-relative URLs are not allowed');\n    }\n\n    try {\n      // Decode URL to prevent encoding-based bypasses\n      let decodedUrl = url;\n      try {\n        decodedUrl = decodeURIComponent(url);\n      } catch {\n        // If decoding fails, use original URL\n      }\n      \n      const parsed = new URL(decodedUrl);\n      \n      // Protocol validation\n      if (!['http:', 'https:'].includes(parsed.protocol)) {\n        throw new Error('Only HTTP(S) URLs are allowed');\n      }\n      \n      // Enhanced SSRF protection with IDN normalization\n      let hostname = parsed.hostname.toLowerCase();\n      \n      // Handle IDN (International Domain Names) by converting to ASCII\n      try {\n        const idnNormalized = new URL(`http://${hostname}`).hostname;\n        hostname = idnNormalized;\n      } catch (idnError) {\n        // If IDN conversion fails, reject the URL for security\n        throw new Error('Invalid hostname: IDN conversion failed - potentially malicious domain name');\n      }\n      \n      // Check for private IPs (now with IDN-normalized hostname)\n      if (this.isPrivateIP(hostname)) {\n        throw new Error('Private network URLs are not allowed');\n      }\n      \n      // Additional SSRF checks for encoded IPs\n      if (this.isEncodedPrivateIP(hostname)) {\n        throw new Error('Encoded private network URLs are not allowed');\n      }\n\n      return url;\n    } catch (error) {\n      if (error instanceof Error && (error.message.includes('Private network') || error.message.includes('Encoded private'))) {\n        throw error;\n      }\n      const errorMessage = error instanceof Error ? error.message : 'Unknown error';\n      throw new Error(`Invalid URL format: ${errorMessage}`);\n    }\n  }\n\n  /**\n   * Validate expiry days for sharing\n   */\n  static validateExpiryDays(days: number): number {\n    if (typeof days !== 'number') {\n      throw new Error('Expiry days must be a valid number');\n    }\n    \n    if (isNaN(days) || !isFinite(days)) {\n      throw new Error('Expiry days must be a valid number');\n    }\n\n    if (days < 1 || days > 365) {\n      throw new Error('Expiry days must be between 1 and 365');\n    }\n\n    return Math.floor(days);\n  }\n\n  /**\n   * Validate boolean confirmation parameters\n   */\n  static validateConfirmation(confirm: boolean, operationName: string): boolean {\n    if (typeof confirm !== 'boolean') {\n      throw new Error(`${operationName} confirmation must be a boolean value`);\n    }\n\n    if (!confirm) {\n      throw new Error(`${operationName} operation requires explicit confirmation (true)`);\n    }\n\n    return confirm;\n  }\n\n  /**\n   * Validate field name for edit operations\n   */\n  static validateEditField(field: string): string {\n    if (!field || typeof field !== 'string') {\n      throw new Error('Field name must be a non-empty string');\n    }\n\n    const validFields = [\n      'name', 'description', 'category', 'instructions', \n      'triggers', 'version', 'author', 'tags'\n    ];\n\n    const normalizedField = field.toLowerCase().trim();\n    if (!validFields.includes(normalizedField)) {\n      throw new Error(`Invalid field name. Must be one of: ${validFields.join(', ')}`);\n    }\n\n    return normalizedField;\n  }\n\n  /**\n   * Check if hostname is a private IP address (IPv4 and IPv6)\n   */\n  private static isPrivateIP(hostname: string): boolean {\n    // Check for localhost variations\n    if (['localhost', '127.0.0.1', '::1'].includes(hostname)) {\n      return true;\n    }\n\n    // Check for private IPv4 ranges\n    const ipv4Match = hostname.match(IPV4_REGEX);\n    \n    if (ipv4Match) {\n      const [, a, b, c, d] = ipv4Match.map(Number);\n      \n      // 10.0.0.0/8\n      if (a === 10) return true;\n      \n      // 172.16.0.0/12\n      if (a === 172 && b >= 16 && b <= 31) return true;\n      \n      // 192.168.0.0/16\n      if (a === 192 && b === 168) return true;\n      \n      // 169.254.0.0/16 (link-local)\n      if (a === 169 && b === 254) return true;\n    }\n\n    // Check for private IPv6 ranges\n    const ipv6Lower = hostname.toLowerCase();\n    \n    // fc00::/7 - Unique Local Addresses (ULA)\n    if (ipv6Lower.startsWith('fc') || ipv6Lower.startsWith('fd')) {\n      return true;\n    }\n    \n    // fe80::/10 - Link-Local Addresses\n    // IPv6 link-local addresses are fe80::/10, meaning the valid range is fe80 through febf\n    const fe80Range = parseInt(ipv6Lower.substring(0, 4), 16);\n    if (fe80Range >= 0xfe80 && fe80Range <= 0xfebf) {\n      return true;\n    }\n    \n    // Additional IPv6 localhost formats\n    if (['::1', '0:0:0:0:0:0:0:1'].includes(ipv6Lower)) {\n      return true;\n    }\n\n    return false;\n  }\n\n  /**\n   * Check for encoded private IP addresses that could bypass basic detection\n   */\n  private static isEncodedPrivateIP(hostname: string): boolean {\n    // Check for decimal encoded IPs (e.g., 2130706433 = 127.0.0.1)\n    if (DECIMAL_IP_REGEX.test(hostname)) {\n      const num = parseInt(hostname, 10);\n      if (num >= 0 && num <= 4294967295) { // Valid IPv4 range\n        // Convert to IP format and check if private\n        const ip = [(num >>> 24) & 255, (num >>> 16) & 255, (num >>> 8) & 255, num & 255].join('.');\n        return this.isPrivateIP(ip);\n      }\n    }\n    \n    // Check for hex encoded IPs (e.g., 0x7f000001 = 127.0.0.1)\n    if (HEX_IP_REGEX.test(hostname)) {\n      const num = parseInt(hostname, 16);\n      if (num >= 0 && num <= 4294967295) {\n        const ip = [(num >>> 24) & 255, (num >>> 16) & 255, (num >>> 8) & 255, num & 255].join('.');\n        return this.isPrivateIP(ip);\n      }\n    }\n    \n    // Check for octal encoded IPs (e.g., 017700000001 = 127.0.0.1)\n    if (OCTAL_IP_REGEX.test(hostname)) {\n      const num = parseInt(hostname, 8);\n      if (num >= 0 && num <= 4294967295) {\n        const ip = [(num >>> 24) & 255, (num >>> 16) & 255, (num >>> 8) & 255, num & 255].join('.');\n        return this.isPrivateIP(ip);\n      }\n    }\n    \n    return false;\n  }\n}\n\n/**\n * Validate and sanitize a filename\n */\nexport function validateFilename(filename: string): string {\n  if (!filename || typeof filename !== 'string') {\n    throw new Error('Filename must be a non-empty string');\n  }\n  \n  if (filename.length > SECURITY_LIMITS.MAX_FILENAME_LENGTH) {\n    throw new Error(`Filename too long (max ${SECURITY_LIMITS.MAX_FILENAME_LENGTH} characters)`);\n  }\n  \n  // Remove any path separators and dangerous characters\n  const sanitized = filename.replace(FILENAME_DANGEROUS_REGEX, '').replace(FILENAME_LEADING_DOTS_REGEX, '');\n  \n  if (!RegexValidator.validate(sanitized, VALIDATION_PATTERNS.SAFE_FILENAME, { maxLength: SECURITY_LIMITS.MAX_FILENAME_LENGTH })) {\n    throw new Error('Invalid filename format. Use alphanumeric characters, hyphens, underscores, and dots only.');\n  }\n  \n  return sanitized;\n}\n\n/**\n * Validate and sanitize a path\n */\nexport function validatePath(inputPath: string, baseDir?: string): string {\n  if (!inputPath || typeof inputPath !== 'string') {\n    throw new Error('Path must be a non-empty string');\n  }\n  \n  // If baseDir is provided and inputPath is absolute, reject it\n  // Check both Unix-style and Windows-style absolute paths for cross-platform security\n  const isUnixAbsolute = path.isAbsolute(inputPath);\n  const isWindowsAbsolute = /^[a-zA-Z]:[\\\\/]/.test(inputPath);\n  \n  if (baseDir && (isUnixAbsolute || isWindowsAbsolute)) {\n    throw new Error('Absolute paths not allowed when base directory is specified');\n  }\n  \n  // Remove leading/trailing slashes and normalize\n  // Length limits added to prevent ReDoS attacks\n  // WINDOWS FIX: Convert backslashes to forward slashes for cross-platform compatibility\n  let normalized = inputPath.replace(/\\\\/g, '/');\n  \n  // FIX: Preserve leading slash for absolute paths\n  const isAbsolute = normalized.startsWith('/') || isWindowsAbsolute;\n  \n  // Remove trailing slashes and normalize multiple slashes\n  normalized = normalized.replace(/\\/{1,100}$/g, '').replace(/\\/{2,100}/g, '/');\n  \n  // Preserve the leading slash if it was an absolute path\n  if (isAbsolute && !normalized.startsWith('/') && !isWindowsAbsolute) {\n    normalized = '/' + normalized;\n  }\n  \n  if (!VALIDATION_PATTERNS.SAFE_PATH.test(normalized)) {\n    throw new Error('Invalid path format. Use alphanumeric characters, hyphens, underscores, dots, and forward slashes only.');\n  }\n  \n  // Check for path traversal attempts\n  if (normalized.includes('..') || normalized.includes('./') || normalized.includes('/.')) {\n    throw new Error('Path traversal not allowed');\n  }\n  \n  // Validate path depth\n  const depth = normalized.split('/').length;\n  if (depth > SECURITY_LIMITS.MAX_PATH_DEPTH) {\n    throw new Error(`Path too deep (max ${SECURITY_LIMITS.MAX_PATH_DEPTH} levels)`);\n  }\n  \n  // If baseDir provided, ensure path is within it\n  if (baseDir) {\n    const resolvedPath = path.resolve(baseDir, normalized);\n    const resolvedBase = path.resolve(baseDir);\n    \n    if (!resolvedPath.startsWith(resolvedBase)) {\n      throw new Error('Path traversal attempt detected');\n    }\n  }\n  \n  return normalized;\n}\n\n/**\n * Validate and sanitize a username\n */\nexport function validateUsername(username: string): string {\n  if (!username || typeof username !== 'string') {\n    throw new Error('Username must be a non-empty string');\n  }\n  \n  if (!VALIDATION_PATTERNS.SAFE_USERNAME.test(username)) {\n    throw new Error('Invalid username format. Use alphanumeric characters, hyphens, underscores, and dots only.');\n  }\n  \n  return username.toLowerCase();\n}\n\n/**\n * Validate a category\n */\nexport function validateCategory(category: string): string {\n  if (!category || typeof category !== 'string') {\n    throw new Error('Category must be a non-empty string');\n  }\n  \n  if (!RegexValidator.validate(category, VALIDATION_PATTERNS.SAFE_CATEGORY, { maxLength: 50 })) {\n    throw new Error('Invalid category format. Use alphabetic characters, hyphens, and underscores only.');\n  }\n  \n  const normalized = category.toLowerCase();\n  \n  if (!VALID_CATEGORIES.includes(normalized)) {\n    throw new Error(`Invalid category. Must be one of: ${VALID_CATEGORIES.join(', ')}`);\n  }\n  \n  return normalized;\n}\n\n/**\n * Validate content size\n */\nexport function validateContentSize(content: string, maxSize: number = SECURITY_LIMITS.MAX_CONTENT_LENGTH): void {\n  if (!content || typeof content !== 'string') {\n    throw new Error('Content must be a non-empty string');\n  }\n  \n  const sizeBytes = Buffer.byteLength(content, 'utf8');\n  if (sizeBytes > maxSize) {\n    throw new Error(`Content too large (${sizeBytes} bytes, max ${maxSize} bytes)`);\n  }\n}\n\n/**\n * Comprehensive input validation before pattern matching\n * Validates all content types with appropriate limits\n */\nexport interface ContentValidationOptions {\n  maxContentLength?: number;\n  maxYamlLength?: number;\n  maxMetadataFieldLength?: number;\n  maxFileSize?: number;\n}\n\nexport function validateInputLengths(\n  content: string,\n  contentType: 'full' | 'yaml' | 'metadata' | 'field',\n  options: ContentValidationOptions = {}\n): void {\n  const limits = {\n    maxContentLength: options.maxContentLength ?? SECURITY_LIMITS.MAX_CONTENT_LENGTH,\n    maxYamlLength: options.maxYamlLength ?? SECURITY_LIMITS.MAX_YAML_LENGTH,\n    maxMetadataFieldLength: options.maxMetadataFieldLength ?? SECURITY_LIMITS.MAX_METADATA_FIELD_LENGTH,\n    maxFileSize: options.maxFileSize ?? SECURITY_LIMITS.MAX_FILE_SIZE\n  };\n\n  // Validate based on content type\n  switch (contentType) {\n    case 'full':\n      if (content.length > limits.maxContentLength) {\n        throw new Error(\n          `Content exceeds maximum length of ${limits.maxContentLength} characters (${content.length} provided)`\n        );\n      }\n      break;\n    \n    case 'yaml':\n      if (content.length > limits.maxYamlLength) {\n        throw new Error(\n          `YAML content exceeds maximum length of ${limits.maxYamlLength} characters (${content.length} provided)`\n        );\n      }\n      break;\n    \n    case 'metadata':\n      // For metadata, check overall size\n      if (content.length > limits.maxYamlLength) {\n        throw new Error(\n          `Metadata exceeds maximum length of ${limits.maxYamlLength} characters (${content.length} provided)`\n        );\n      }\n      break;\n    \n    case 'field':\n      if (content.length > limits.maxMetadataFieldLength) {\n        throw new Error(\n          `Field exceeds maximum length of ${limits.maxMetadataFieldLength} characters (${content.length} provided)`\n        );\n      }\n      break;\n  }\n}\n\n/**\n * General input sanitization\n */\nexport function sanitizeInput(input: string, maxLength: number = 1000): string {\n  if (!input || typeof input !== 'string') {\n    return '';\n  }\n  \n  // Remove potentially dangerous characters and limit length\n  return input\n    .replace(CONTROL_CHARS_REGEX, '') // Remove control characters\n    .replace(HTML_DANGEROUS_REGEX, '') // Remove HTML-dangerous characters\n    .replace(SHELL_METACHAR_REGEX, '') // Remove shell metacharacters (expanded)\n    .replace(RTL_ZEROWIDTH_REGEX, '') // Remove RTL override and zero-width chars\n    .substring(0, maxLength)\n    .trim();\n}"]}
450
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"InputValidator.js","sourceRoot":"","sources":["../../src/security/InputValidator.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Input validation and sanitization functions\n */\n\nimport * as path from 'path';\nimport { SECURITY_LIMITS, VALIDATION_PATTERNS } from './constants.js';\nimport { VALID_CATEGORIES } from '../config/constants.js';\nimport { RegexValidator } from './regexValidator.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\nimport { ValidationErrorCodes } from '../utils/errorCodes.js';\n\n// Pre-compiled regex patterns for better performance\n// These patterns are used repeatedly and benefit from pre-compilation\nconst CONTROL_CHARS_REGEX = /[\\x00-\\x1F\\x7F]/g;\nconst HTML_DANGEROUS_REGEX = /[<>'\"&]/g;\nconst SHELL_METACHAR_REGEX = /[;&|`$()!\\\\~*?{}]/g;\nconst RTL_ZEROWIDTH_REGEX = /[\\u202E\\uFEFF]/g;\nconst COLLECTION_PATH_CHAR_REGEX = /[a-zA-Z0-9\\/\\-_.]/;\nconst VALID_COLLECTION_PATH_REGEX = /^[a-zA-Z0-9\\/\\-_.]*$/;\nconst IPV4_REGEX = /^(\\d+)\\.(\\d+)\\.(\\d+)\\.(\\d+)$/;\nconst DECIMAL_IP_REGEX = /^\\d{8,10}$/;\nconst HEX_IP_REGEX = /^0x[0-9a-f]{1,8}$/i;\nconst OCTAL_IP_REGEX = /^0[0-7]{8,11}$/;\nconst FILENAME_DANGEROUS_REGEX = /[\\/\\\\:*?\"<>|]/g;\nconst FILENAME_LEADING_DOTS_REGEX = /^\\.+/;\nconst PATH_NORMALIZE_REGEX = /^\\/{1,100}|\\/{1,100}$/g;\nconst PATH_MULTIPLE_SLASHES_REGEX = /\\/{1,100}/g;\nconst URL_PLUS_DECODE_REGEX = /\\+/g;\n\n/**\n * Enhanced input validation for MCP tools\n */\nexport class MCPInputValidator {\n  /**\n   * Validate a persona identifier (name or filename)\n   */\n  static validatePersonaIdentifier(identifier: string): string {\n    if (!identifier || typeof identifier !== 'string') {\n      throw ErrorHandler.createError('Persona identifier must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PERSONA_ID);\n    }\n\n    if (identifier.length > 100) {\n      throw ErrorHandler.createError('Persona identifier too long (max 100 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);\n    }\n\n    // Allow persona names and filenames\n    const sanitized = sanitizeInput(identifier, 100);\n    if (!sanitized) {\n      throw ErrorHandler.createError('Persona identifier contains only invalid characters', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PERSONA_ID);\n    }\n\n    return sanitized;\n  }\n\n  /**\n   * Validate search query for collection\n   */\n  static validateSearchQuery(query: string): string {\n    if (!query || typeof query !== 'string') {\n      throw ErrorHandler.createError('Search query must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_SEARCH_QUERY);\n    }\n\n    if (query.length < 2) {\n      throw ErrorHandler.createError('Search query too short (minimum 2 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);\n    }\n\n    if (query.length > 200) {\n      throw ErrorHandler.createError('Search query too long (max 200 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);\n    }\n\n    // Sanitize but preserve spaces for search\n    const sanitized = query\n      .replace(CONTROL_CHARS_REGEX, '') // Remove control characters\n      .replace(HTML_DANGEROUS_REGEX, '') // Remove HTML-dangerous characters\n      .replace(SHELL_METACHAR_REGEX, '') // Remove shell metacharacters (expanded)\n      .replace(RTL_ZEROWIDTH_REGEX, '') // Remove RTL override and zero-width chars\n      .trim();\n\n    if (!sanitized) {\n      throw ErrorHandler.createError('Search query contains only invalid characters', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_SEARCH_QUERY);\n    }\n\n    return sanitized;\n  }\n\n  /**\n   * Validate collection path\n   */\n  static validateCollectionPath(path: string): string {\n    if (!path || typeof path !== 'string') {\n      throw ErrorHandler.createError('Collection path must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_COLLECTION_PATH);\n    }\n\n    if (path.length > 500) {\n      throw ErrorHandler.createError('Collection path too long (max 500 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);\n    }\n\n    // GitHub API paths should be safe filename patterns\n    // Use single regex test for better performance (avoids O(n) character-by-character check)\n    if (!VALID_COLLECTION_PATH_REGEX.test(path)) {\n      // Only do character-by-character check if validation fails, to provide detailed error message\n      for (let i = 0; i < path.length; i++) {\n        const char = path[i];\n        if (!COLLECTION_PATH_CHAR_REGEX.test(char)) {\n          throw ErrorHandler.createError(`Invalid character '${char}' in collection path at position ${i + 1}`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_CHARACTER);\n        }\n      }\n      // Fallback error if we somehow don't find the invalid character\n      throw ErrorHandler.createError('Invalid characters in collection path', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_COLLECTION_PATH);\n    }\n\n    // Prevent path traversal in GitHub paths (comprehensive check)\n    const pathLower = path.toLowerCase();\n    const encodedPath = decodeURIComponent(path.replace(URL_PLUS_DECODE_REGEX, ' ')); // Decode URL encoding\n    \n    // Check for various path traversal patterns\n    const traversalPatterns = [\n      '..',          // Basic traversal\n      './',          // Current directory\n      '/../',        // Directory traversal with slashes\n      '\\\\',          // Backslash (Windows-style)\n      '%2e%2e',      // URL-encoded ..\n      '%2e%2e%2f',   // URL-encoded ../\n      '%2e%2e%5c',   // URL-encoded ..\\\n      '%252e%252e',  // Double URL-encoded ..\n      '..%2f',       // Mixed encoding\n      '..%5c',       // Mixed encoding with backslash\n      '..../',       // Dotdot bypass attempt\n      '..;/',        // Semicolon bypass attempt\n    ];\n    \n    for (const pattern of traversalPatterns) {\n      if (pathLower.includes(pattern) || encodedPath.toLowerCase().includes(pattern)) {\n        throw ErrorHandler.createError('Path traversal not allowed in collection path', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.PATH_TRAVERSAL);\n      }\n    }\n\n    return path;\n  }\n\n  /**\n   * Validate URL for import operations\n   */\n  static validateImportUrl(url: string): string {\n    if (!url || typeof url !== 'string') {\n      throw ErrorHandler.createError('URL must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);\n    }\n\n    if (url.length > 2000) {\n      throw ErrorHandler.createError('URL too long (max 2000 characters)', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);\n    }\n\n    // Reject protocol-relative URLs that could bypass validation\n    if (url.startsWith('//')) {\n      throw ErrorHandler.createError('Protocol-relative URLs are not allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);\n    }\n\n    try {\n      // Decode URL to prevent encoding-based bypasses\n      let decodedUrl = url;\n      try {\n        decodedUrl = decodeURIComponent(url);\n      } catch {\n        // If decoding fails, use original URL\n      }\n      \n      const parsed = new URL(decodedUrl);\n      \n      // Protocol validation\n      if (!['http:', 'https:'].includes(parsed.protocol)) {\n        throw ErrorHandler.createError('Only HTTP(S) URLs are allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);\n      }\n      \n      // Enhanced SSRF protection with IDN normalization\n      let hostname = parsed.hostname.toLowerCase();\n      \n      // Handle IDN (International Domain Names) by converting to ASCII\n      try {\n        const idnNormalized = new URL(`http://${hostname}`).hostname;\n        hostname = idnNormalized;\n      } catch (idnError) {\n        // If IDN conversion fails, reject the URL for security\n        throw ErrorHandler.createError('Invalid hostname: IDN conversion failed - potentially malicious domain name', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);\n      }\n      \n      // Check for private IPs (now with IDN-normalized hostname)\n      if (this.isPrivateIP(hostname)) {\n        throw ErrorHandler.createError('Private network URLs are not allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);\n      }\n      \n      // Additional SSRF checks for encoded IPs\n      if (this.isEncodedPrivateIP(hostname)) {\n        throw ErrorHandler.createError('Encoded private network URLs are not allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);\n      }\n\n      return url;\n    } catch (error) {\n      if (error instanceof Error && (error.message.includes('Private network') || error.message.includes('Encoded private'))) {\n        throw error;\n      }\n      const errorMessage = error instanceof Error ? error.message : 'Unknown error';\n      throw ErrorHandler.createError(`Invalid URL format: ${errorMessage}`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_URL);\n    }\n  }\n\n  /**\n   * Validate expiry days for sharing\n   */\n  static validateExpiryDays(days: number): number {\n    if (typeof days !== 'number') {\n      throw ErrorHandler.createError('Expiry days must be a valid number', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_NUMBER);\n    }\n    \n    if (isNaN(days) || !isFinite(days)) {\n      throw ErrorHandler.createError('Expiry days must be a valid number', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_NUMBER);\n    }\n\n    if (days < 1 || days > 365) {\n      throw ErrorHandler.createError('Expiry days must be between 1 and 365', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_RANGE);\n    }\n\n    return Math.floor(days);\n  }\n\n  /**\n   * Validate boolean confirmation parameters\n   */\n  static validateConfirmation(confirm: boolean, operationName: string): boolean {\n    if (typeof confirm !== 'boolean') {\n      throw ErrorHandler.createError(`${operationName} confirmation must be a boolean value`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_TYPE);\n    }\n\n    if (!confirm) {\n      throw ErrorHandler.createError(`${operationName} operation requires explicit confirmation (true)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.CONFIRMATION_REQUIRED);\n    }\n\n    return confirm;\n  }\n\n  /**\n   * Validate field name for edit operations\n   */\n  static validateEditField(field: string): string {\n    if (!field || typeof field !== 'string') {\n      throw ErrorHandler.createError('Field name must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.REQUIRED_FIELD);\n    }\n\n    const validFields = [\n      'name', 'description', 'category', 'instructions', \n      'triggers', 'version', 'author', 'tags'\n    ];\n\n    const normalizedField = field.toLowerCase().trim();\n    if (!validFields.includes(normalizedField)) {\n      throw ErrorHandler.createError(`Invalid field name. Must be one of: ${validFields.join(', ')}`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_INPUT);\n    }\n\n    return normalizedField;\n  }\n\n  /**\n   * Check if hostname is a private IP address (IPv4 and IPv6)\n   */\n  private static isPrivateIP(hostname: string): boolean {\n    // Check for localhost variations\n    if (['localhost', '127.0.0.1', '::1'].includes(hostname)) {\n      return true;\n    }\n\n    // Check for private IPv4 ranges\n    const ipv4Match = hostname.match(IPV4_REGEX);\n    \n    if (ipv4Match) {\n      const [, a, b, c, d] = ipv4Match.map(Number);\n      \n      // 10.0.0.0/8\n      if (a === 10) return true;\n      \n      // 172.16.0.0/12\n      if (a === 172 && b >= 16 && b <= 31) return true;\n      \n      // 192.168.0.0/16\n      if (a === 192 && b === 168) return true;\n      \n      // 169.254.0.0/16 (link-local)\n      if (a === 169 && b === 254) return true;\n    }\n\n    // Check for private IPv6 ranges\n    const ipv6Lower = hostname.toLowerCase();\n    \n    // fc00::/7 - Unique Local Addresses (ULA)\n    if (ipv6Lower.startsWith('fc') || ipv6Lower.startsWith('fd')) {\n      return true;\n    }\n    \n    // fe80::/10 - Link-Local Addresses\n    // IPv6 link-local addresses are fe80::/10, meaning the valid range is fe80 through febf\n    const fe80Range = parseInt(ipv6Lower.substring(0, 4), 16);\n    if (fe80Range >= 0xfe80 && fe80Range <= 0xfebf) {\n      return true;\n    }\n    \n    // Additional IPv6 localhost formats\n    if (['::1', '0:0:0:0:0:0:0:1'].includes(ipv6Lower)) {\n      return true;\n    }\n\n    return false;\n  }\n\n  /**\n   * Check for encoded private IP addresses that could bypass basic detection\n   */\n  private static isEncodedPrivateIP(hostname: string): boolean {\n    // Check for decimal encoded IPs (e.g., 2130706433 = 127.0.0.1)\n    if (DECIMAL_IP_REGEX.test(hostname)) {\n      const num = parseInt(hostname, 10);\n      if (num >= 0 && num <= 4294967295) { // Valid IPv4 range\n        // Convert to IP format and check if private\n        const ip = [(num >>> 24) & 255, (num >>> 16) & 255, (num >>> 8) & 255, num & 255].join('.');\n        return this.isPrivateIP(ip);\n      }\n    }\n    \n    // Check for hex encoded IPs (e.g., 0x7f000001 = 127.0.0.1)\n    if (HEX_IP_REGEX.test(hostname)) {\n      const num = parseInt(hostname, 16);\n      if (num >= 0 && num <= 4294967295) {\n        const ip = [(num >>> 24) & 255, (num >>> 16) & 255, (num >>> 8) & 255, num & 255].join('.');\n        return this.isPrivateIP(ip);\n      }\n    }\n    \n    // Check for octal encoded IPs (e.g., 017700000001 = 127.0.0.1)\n    if (OCTAL_IP_REGEX.test(hostname)) {\n      const num = parseInt(hostname, 8);\n      if (num >= 0 && num <= 4294967295) {\n        const ip = [(num >>> 24) & 255, (num >>> 16) & 255, (num >>> 8) & 255, num & 255].join('.');\n        return this.isPrivateIP(ip);\n      }\n    }\n    \n    return false;\n  }\n}\n\n/**\n * Validate and sanitize a filename\n */\nexport function validateFilename(filename: string): string {\n  if (!filename || typeof filename !== 'string') {\n    throw ErrorHandler.createError('Filename must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_FILENAME);\n  }\n  \n  if (filename.length > SECURITY_LIMITS.MAX_FILENAME_LENGTH) {\n    throw ErrorHandler.createError(`Filename too long (max ${SECURITY_LIMITS.MAX_FILENAME_LENGTH} characters)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_LENGTH);\n  }\n  \n  // Remove any path separators and dangerous characters\n  const sanitized = filename.replace(FILENAME_DANGEROUS_REGEX, '').replace(FILENAME_LEADING_DOTS_REGEX, '');\n  \n  if (!RegexValidator.validate(sanitized, VALIDATION_PATTERNS.SAFE_FILENAME, { maxLength: SECURITY_LIMITS.MAX_FILENAME_LENGTH })) {\n    throw ErrorHandler.createError('Invalid filename format. Use alphanumeric characters, hyphens, underscores, and dots only.', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_FILENAME);\n  }\n  \n  return sanitized;\n}\n\n/**\n * Validate and sanitize a path\n */\nexport function validatePath(inputPath: string, baseDir?: string): string {\n  if (!inputPath || typeof inputPath !== 'string') {\n    throw ErrorHandler.createError('Path must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PATH);\n  }\n  \n  // If baseDir is provided and inputPath is absolute, reject it\n  // Check both Unix-style and Windows-style absolute paths for cross-platform security\n  const isUnixAbsolute = path.isAbsolute(inputPath);\n  const isWindowsAbsolute = /^[a-zA-Z]:[\\\\/]/.test(inputPath);\n  \n  if (baseDir && (isUnixAbsolute || isWindowsAbsolute)) {\n    throw ErrorHandler.createError('Absolute paths not allowed when base directory is specified', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PATH);\n  }\n  \n  // Remove leading/trailing slashes and normalize\n  // Length limits added to prevent ReDoS attacks\n  // WINDOWS FIX: Convert backslashes to forward slashes for cross-platform compatibility\n  let normalized = inputPath.replace(/\\\\/g, '/');\n  \n  // FIX: Preserve leading slash for absolute paths\n  const isAbsolute = normalized.startsWith('/') || isWindowsAbsolute;\n  \n  // Remove trailing slashes and normalize multiple slashes\n  normalized = normalized.replace(/\\/{1,100}$/g, '').replace(/\\/{2,100}/g, '/');\n  \n  // Preserve the leading slash if it was an absolute path\n  if (isAbsolute && !normalized.startsWith('/') && !isWindowsAbsolute) {\n    normalized = '/' + normalized;\n  }\n  \n  if (!VALIDATION_PATTERNS.SAFE_PATH.test(normalized)) {\n    throw ErrorHandler.createError('Invalid path format. Use alphanumeric characters, hyphens, underscores, dots, and forward slashes only.', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PATH);\n  }\n  \n  // Check for path traversal attempts\n  if (normalized.includes('..') || normalized.includes('./') || normalized.includes('/.')) {\n    throw ErrorHandler.createError('Path traversal not allowed', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.PATH_TRAVERSAL);\n  }\n  \n  // Validate path depth\n  const depth = normalized.split('/').length;\n  if (depth > SECURITY_LIMITS.MAX_PATH_DEPTH) {\n    throw ErrorHandler.createError(`Path too deep (max ${SECURITY_LIMITS.MAX_PATH_DEPTH} levels)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_PATH);\n  }\n  \n  // If baseDir provided, ensure path is within it\n  if (baseDir) {\n    const resolvedPath = path.resolve(baseDir, normalized);\n    const resolvedBase = path.resolve(baseDir);\n    \n    if (!resolvedPath.startsWith(resolvedBase)) {\n      throw ErrorHandler.createError('Path traversal attempt detected', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.PATH_TRAVERSAL);\n    }\n  }\n  \n  return normalized;\n}\n\n/**\n * Validate and sanitize a username\n */\nexport function validateUsername(username: string): string {\n  if (!username || typeof username !== 'string') {\n    throw ErrorHandler.createError('Username must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.REQUIRED_FIELD);\n  }\n  \n  if (!VALIDATION_PATTERNS.SAFE_USERNAME.test(username)) {\n    throw ErrorHandler.createError('Invalid username format. Use alphanumeric characters, hyphens, underscores, and dots only.', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_FORMAT);\n  }\n  \n  return username.toLowerCase();\n}\n\n/**\n * Validate a category\n */\nexport function validateCategory(category: string): string {\n  if (!category || typeof category !== 'string') {\n    throw ErrorHandler.createError('Category must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_CATEGORY);\n  }\n  \n  if (!RegexValidator.validate(category, VALIDATION_PATTERNS.SAFE_CATEGORY, { maxLength: 50 })) {\n    throw ErrorHandler.createError('Invalid category format. Use alphabetic characters, hyphens, and underscores only.', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_CATEGORY);\n  }\n  \n  const normalized = category.toLowerCase();\n  \n  if (!VALID_CATEGORIES.includes(normalized)) {\n    throw ErrorHandler.createError(`Invalid category. Must be one of: ${VALID_CATEGORIES.join(', ')}`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.INVALID_CATEGORY);\n  }\n  \n  return normalized;\n}\n\n/**\n * Validate content size\n */\nexport function validateContentSize(content: string, maxSize: number = SECURITY_LIMITS.MAX_CONTENT_LENGTH): void {\n  if (!content || typeof content !== 'string') {\n    throw ErrorHandler.createError('Content must be a non-empty string', ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.REQUIRED_FIELD);\n  }\n  \n  const sizeBytes = Buffer.byteLength(content, 'utf8');\n  if (sizeBytes > maxSize) {\n    throw ErrorHandler.createError(`Content too large (${sizeBytes} bytes, max ${maxSize} bytes)`, ErrorCategory.VALIDATION_ERROR, ValidationErrorCodes.CONTENT_TOO_LARGE);\n  }\n}\n\n/**\n * Comprehensive input validation before pattern matching\n * Validates all content types with appropriate limits\n */\nexport interface ContentValidationOptions {\n  maxContentLength?: number;\n  maxYamlLength?: number;\n  maxMetadataFieldLength?: number;\n  maxFileSize?: number;\n}\n\nexport function validateInputLengths(\n  content: string,\n  contentType: 'full' | 'yaml' | 'metadata' | 'field',\n  options: ContentValidationOptions = {}\n): void {\n  const limits = {\n    maxContentLength: options.maxContentLength ?? SECURITY_LIMITS.MAX_CONTENT_LENGTH,\n    maxYamlLength: options.maxYamlLength ?? SECURITY_LIMITS.MAX_YAML_LENGTH,\n    maxMetadataFieldLength: options.maxMetadataFieldLength ?? SECURITY_LIMITS.MAX_METADATA_FIELD_LENGTH,\n    maxFileSize: options.maxFileSize ?? SECURITY_LIMITS.MAX_FILE_SIZE\n  };\n\n  // Validate based on content type\n  switch (contentType) {\n    case 'full':\n      if (content.length > limits.maxContentLength) {\n        throw ErrorHandler.createError(\n          `Content exceeds maximum length of ${limits.maxContentLength} characters (${content.length} provided)`,\n          ErrorCategory.VALIDATION_ERROR,\n          ValidationErrorCodes.CONTENT_TOO_LARGE\n        );\n      }\n      break;\n    \n    case 'yaml':\n      if (content.length > limits.maxYamlLength) {\n        throw ErrorHandler.createError(\n          `YAML content exceeds maximum length of ${limits.maxYamlLength} characters (${content.length} provided)`,\n          ErrorCategory.VALIDATION_ERROR,\n          ValidationErrorCodes.CONTENT_TOO_LARGE\n        );\n      }\n      break;\n    \n    case 'metadata':\n      // For metadata, check overall size\n      if (content.length > limits.maxYamlLength) {\n        throw ErrorHandler.createError(\n          `Metadata exceeds maximum length of ${limits.maxYamlLength} characters (${content.length} provided)`,\n          ErrorCategory.VALIDATION_ERROR,\n          ValidationErrorCodes.CONTENT_TOO_LARGE\n        );\n      }\n      break;\n    \n    case 'field':\n      if (content.length > limits.maxMetadataFieldLength) {\n        throw ErrorHandler.createError(\n          `Field exceeds maximum length of ${limits.maxMetadataFieldLength} characters (${content.length} provided)`,\n          ErrorCategory.VALIDATION_ERROR,\n          ValidationErrorCodes.CONTENT_TOO_LARGE\n        );\n      }\n      break;\n  }\n}\n\n/**\n * General input sanitization\n */\nexport function sanitizeInput(input: string, maxLength: number = 1000): string {\n  if (!input || typeof input !== 'string') {\n    return '';\n  }\n  \n  // Remove potentially dangerous characters and limit length\n  return input\n    .replace(CONTROL_CHARS_REGEX, '') // Remove control characters\n    .replace(HTML_DANGEROUS_REGEX, '') // Remove HTML-dangerous characters\n    .replace(SHELL_METACHAR_REGEX, '') // Remove shell metacharacters (expanded)\n    .replace(RTL_ZEROWIDTH_REGEX, '') // Remove RTL override and zero-width chars\n    .substring(0, maxLength)\n    .trim();\n}"]}
@@ -1 +1 @@
1
- {"version":3,"file":"SecurityAuditor.d.ts","sourceRoot":"","sources":["../../../src/security/audit/SecurityAuditor.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAGH,OAAO,KAAK,EACV,mBAAmB,EAEnB,UAAU,EAIX,MAAM,YAAY,CAAC;AAWpB,qBAAa,eAAe;IAC1B,OAAO,CAAC,MAAM,CAAsB;IACpC,OAAO,CAAC,QAAQ,CAAyB;IACzC,OAAO,CAAC,YAAY,CAAuC;gBAE/C,MAAM,EAAE,mBAAmB;IAMvC;;OAEG;IACH,OAAO,CAAC,kBAAkB;IAiB1B;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAYxB;;OAEG;IACG,KAAK,CAAC,WAAW,GAAE,MAAsB,GAAG,OAAO,CAAC,UAAU,CAAC;IA6CrE;;OAEG;IACH,OAAO,CAAC,kBAAkB;IAkE1B;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAsCxB;;OAEG;YACW,eAAe;IA6B7B;;OAEG;IACH,OAAO,CAAC,eAAe;IAoBvB;;OAEG;IACH,MAAM,CAAC,gBAAgB,IAAI,mBAAmB;CAmC/C"}
1
+ {"version":3,"file":"SecurityAuditor.d.ts","sourceRoot":"","sources":["../../../src/security/audit/SecurityAuditor.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAIH,OAAO,KAAK,EACV,mBAAmB,EAEnB,UAAU,EAIX,MAAM,YAAY,CAAC;AAYpB,qBAAa,eAAe;IAC1B,OAAO,CAAC,MAAM,CAAsB;IACpC,OAAO,CAAC,QAAQ,CAAyB;IACzC,OAAO,CAAC,YAAY,CAAuC;gBAE/C,MAAM,EAAE,mBAAmB;IAMvC;;OAEG;IACH,OAAO,CAAC,kBAAkB;IAiB1B;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAYxB;;OAEG;IACG,KAAK,CAAC,WAAW,GAAE,MAAsB,GAAG,OAAO,CAAC,UAAU,CAAC;IA6CrE;;OAEG;IACH,OAAO,CAAC,kBAAkB;IAqE1B;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAsCxB;;OAEG;YACW,eAAe;IA+B7B;;OAEG;IACH,OAAO,CAAC,eAAe;IAoBvB;;OAEG;IACH,MAAM,CAAC,gBAAgB,IAAI,mBAAmB;CAmC/C"}