@dollhousemcp/mcp-server 1.5.2 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/CHANGELOG.md +56 -0
  2. package/README.md +494 -111
  3. package/data/agents/code-reviewer.md +8 -1
  4. package/data/agents/research-assistant.md +8 -1
  5. package/data/agents/task-manager.md +8 -1
  6. package/data/ensembles/business-advisor.md +8 -1
  7. package/data/ensembles/creative-studio.md +8 -1
  8. package/data/ensembles/development-team.md +8 -1
  9. package/data/ensembles/security-analysis-team.md +8 -1
  10. package/data/memories/conversation-history.md +8 -1
  11. package/data/memories/learning-progress.md +8 -1
  12. package/data/memories/project-context.md +8 -1
  13. package/data/personas/business-consultant.md +8 -1
  14. package/data/personas/creative-writer.md +8 -1
  15. package/data/personas/debug-detective.md +8 -1
  16. package/data/personas/eli5-explainer.md +8 -1
  17. package/data/personas/security-analyst.md +8 -1
  18. package/data/personas/technical-analyst.md +8 -1
  19. package/data/skills/code-review.md +8 -1
  20. package/data/skills/creative-writing.md +8 -1
  21. package/data/skills/data-analysis.md +8 -1
  22. package/data/skills/penetration-testing.md +8 -1
  23. package/data/skills/research.md +8 -1
  24. package/data/skills/threat-modeling.md +8 -1
  25. package/data/skills/translation.md +8 -1
  26. package/data/templates/code-documentation.md +8 -1
  27. package/data/templates/email-professional.md +8 -1
  28. package/data/templates/meeting-notes.md +8 -1
  29. package/data/templates/penetration-test-report.md +8 -1
  30. package/data/templates/project-brief.md +8 -1
  31. package/data/templates/report-executive.md +8 -1
  32. package/data/templates/security-vulnerability-report.md +8 -1
  33. package/data/templates/threat-assessment-report.md +8 -1
  34. package/dist/auth/GitHubAuthManager.d.ts +6 -1
  35. package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
  36. package/dist/auth/GitHubAuthManager.js +45 -18
  37. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
  38. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
  39. package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
  40. package/dist/cache/CollectionCache.d.ts.map +1 -1
  41. package/dist/cache/CollectionCache.js +13 -3
  42. package/dist/cache/CollectionIndexCache.d.ts +77 -0
  43. package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
  44. package/dist/cache/CollectionIndexCache.js +349 -0
  45. package/dist/cache/LRUCache.d.ts +93 -0
  46. package/dist/cache/LRUCache.d.ts.map +1 -0
  47. package/dist/cache/LRUCache.js +299 -0
  48. package/dist/cache/index.d.ts +1 -0
  49. package/dist/cache/index.d.ts.map +1 -1
  50. package/dist/cache/index.js +2 -1
  51. package/dist/collection/CollectionBrowser.d.ts +21 -1
  52. package/dist/collection/CollectionBrowser.d.ts.map +1 -1
  53. package/dist/collection/CollectionBrowser.js +130 -10
  54. package/dist/collection/CollectionIndexManager.d.ts +151 -0
  55. package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
  56. package/dist/collection/CollectionIndexManager.js +499 -0
  57. package/dist/collection/CollectionSearch.d.ts +55 -0
  58. package/dist/collection/CollectionSearch.d.ts.map +1 -1
  59. package/dist/collection/CollectionSearch.js +338 -13
  60. package/dist/collection/CollectionSeeder.d.ts.map +1 -1
  61. package/dist/collection/CollectionSeeder.js +38 -1
  62. package/dist/collection/ElementInstaller.d.ts +31 -0
  63. package/dist/collection/ElementInstaller.d.ts.map +1 -1
  64. package/dist/collection/ElementInstaller.js +77 -15
  65. package/dist/collection/PersonaSubmitter.d.ts +1 -1
  66. package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
  67. package/dist/collection/PersonaSubmitter.js +2 -2
  68. package/dist/collection/index.d.ts +1 -0
  69. package/dist/collection/index.d.ts.map +1 -1
  70. package/dist/collection/index.js +2 -1
  71. package/dist/config/ConfigManager.d.ts +78 -0
  72. package/dist/config/ConfigManager.d.ts.map +1 -0
  73. package/dist/config/ConfigManager.js +216 -0
  74. package/dist/config/element-types.d.ts +135 -0
  75. package/dist/config/element-types.d.ts.map +1 -0
  76. package/dist/config/element-types.js +108 -0
  77. package/dist/config/index.d.ts +2 -0
  78. package/dist/config/index.d.ts.map +1 -1
  79. package/dist/config/index.js +3 -1
  80. package/dist/config/portfolio-constants.d.ts +83 -0
  81. package/dist/config/portfolio-constants.d.ts.map +1 -0
  82. package/dist/config/portfolio-constants.js +99 -0
  83. package/dist/elements/BaseElement.d.ts +14 -2
  84. package/dist/elements/BaseElement.d.ts.map +1 -1
  85. package/dist/elements/BaseElement.js +88 -6
  86. package/dist/elements/agents/Agent.d.ts +10 -1
  87. package/dist/elements/agents/Agent.d.ts.map +1 -1
  88. package/dist/elements/agents/Agent.js +66 -19
  89. package/dist/elements/agents/AgentManager.d.ts +2 -0
  90. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  91. package/dist/elements/agents/AgentManager.js +12 -10
  92. package/dist/elements/skills/Skill.d.ts +10 -1
  93. package/dist/elements/skills/Skill.d.ts.map +1 -1
  94. package/dist/elements/skills/Skill.js +40 -3
  95. package/dist/elements/skills/SkillManager.d.ts +1 -0
  96. package/dist/elements/skills/SkillManager.d.ts.map +1 -1
  97. package/dist/elements/skills/SkillManager.js +10 -4
  98. package/dist/elements/templates/Template.d.ts +10 -1
  99. package/dist/elements/templates/Template.d.ts.map +1 -1
  100. package/dist/elements/templates/Template.js +35 -18
  101. package/dist/elements/templates/TemplateManager.d.ts +1 -1
  102. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  103. package/dist/elements/templates/TemplateManager.js +6 -5
  104. package/dist/generated/version.d.ts +2 -2
  105. package/dist/generated/version.js +3 -3
  106. package/dist/index.barrel.d.ts +1 -2
  107. package/dist/index.barrel.d.ts.map +1 -1
  108. package/dist/index.barrel.js +2 -4
  109. package/dist/index.d.ts +143 -25
  110. package/dist/index.d.ts.map +1 -1
  111. package/dist/index.js +1883 -310
  112. package/dist/persona/PersonaElement.d.ts +10 -0
  113. package/dist/persona/PersonaElement.d.ts.map +1 -1
  114. package/dist/persona/PersonaElement.js +55 -32
  115. package/dist/persona/PersonaElementManager.d.ts.map +1 -1
  116. package/dist/persona/PersonaElementManager.js +13 -11
  117. package/dist/persona/PersonaLoader.d.ts.map +1 -1
  118. package/dist/persona/PersonaLoader.js +8 -2
  119. package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
  120. package/dist/persona/export-import/PersonaImporter.js +24 -5
  121. package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
  122. package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
  123. package/dist/persona/export-import/PersonaSharer.js +198 -22
  124. package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
  125. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  126. package/dist/portfolio/DefaultElementProvider.js +499 -7
  127. package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
  128. package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
  129. package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
  130. package/dist/portfolio/MigrationManager.d.ts.map +1 -1
  131. package/dist/portfolio/MigrationManager.js +136 -3
  132. package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
  133. package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
  134. package/dist/portfolio/PortfolioIndexManager.js +478 -0
  135. package/dist/portfolio/PortfolioManager.d.ts +5 -0
  136. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  137. package/dist/portfolio/PortfolioManager.js +61 -20
  138. package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
  139. package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
  140. package/dist/portfolio/PortfolioRepoManager.js +337 -0
  141. package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
  142. package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
  143. package/dist/portfolio/UnifiedIndexManager.js +1434 -0
  144. package/dist/portfolio/index.d.ts +15 -0
  145. package/dist/portfolio/index.d.ts.map +1 -0
  146. package/dist/portfolio/index.js +15 -0
  147. package/dist/portfolio/types.d.ts +7 -0
  148. package/dist/portfolio/types.d.ts.map +1 -1
  149. package/dist/portfolio/types.js +6 -1
  150. package/dist/security/InputValidator.d.ts.map +1 -1
  151. package/dist/security/InputValidator.js +50 -48
  152. package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
  153. package/dist/security/audit/SecurityAuditor.js +17 -9
  154. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  155. package/dist/security/audit/config/suppressions.js +19 -3
  156. package/dist/security/contentValidator.d.ts +2 -0
  157. package/dist/security/contentValidator.d.ts.map +1 -1
  158. package/dist/security/contentValidator.js +115 -4
  159. package/dist/security/secureYamlParser.d.ts +1 -0
  160. package/dist/security/secureYamlParser.d.ts.map +1 -1
  161. package/dist/security/secureYamlParser.js +29 -7
  162. package/dist/security/securityMonitor.d.ts +1 -1
  163. package/dist/security/securityMonitor.d.ts.map +1 -1
  164. package/dist/security/securityMonitor.js +1 -1
  165. package/dist/security/tokenManager.d.ts +1 -1
  166. package/dist/security/tokenManager.d.ts.map +1 -1
  167. package/dist/security/tokenManager.js +30 -10
  168. package/dist/server/ServerSetup.d.ts +22 -2
  169. package/dist/server/ServerSetup.d.ts.map +1 -1
  170. package/dist/server/ServerSetup.js +77 -12
  171. package/dist/server/tools/AuthTools.d.ts.map +1 -1
  172. package/dist/server/tools/AuthTools.js +33 -1
  173. package/dist/server/tools/BuildInfoTools.d.ts +25 -0
  174. package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
  175. package/dist/server/tools/BuildInfoTools.js +36 -0
  176. package/dist/server/tools/CollectionTools.d.ts.map +1 -1
  177. package/dist/server/tools/CollectionTools.js +55 -46
  178. package/dist/server/tools/ConfigTools.d.ts.map +1 -1
  179. package/dist/server/tools/ConfigTools.js +29 -1
  180. package/dist/server/tools/PersonaTools.d.ts +4 -2
  181. package/dist/server/tools/PersonaTools.d.ts.map +1 -1
  182. package/dist/server/tools/PersonaTools.js +5 -152
  183. package/dist/server/tools/PortfolioTools.d.ts +12 -0
  184. package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
  185. package/dist/server/tools/PortfolioTools.js +221 -0
  186. package/dist/server/tools/index.d.ts +3 -1
  187. package/dist/server/tools/index.d.ts.map +1 -1
  188. package/dist/server/tools/index.js +4 -2
  189. package/dist/server/types.d.ts +40 -5
  190. package/dist/server/types.d.ts.map +1 -1
  191. package/dist/server/types.js +1 -1
  192. package/dist/services/BuildInfoService.d.ts +84 -0
  193. package/dist/services/BuildInfoService.d.ts.map +1 -0
  194. package/dist/services/BuildInfoService.js +271 -0
  195. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
  196. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
  197. package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
  198. package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
  199. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
  200. package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
  201. package/dist/tools/portfolio/types.d.ts +41 -0
  202. package/dist/tools/portfolio/types.d.ts.map +1 -0
  203. package/dist/tools/portfolio/types.js +15 -0
  204. package/dist/types/collection.d.ts +51 -0
  205. package/dist/types/collection.d.ts.map +1 -1
  206. package/dist/types/collection.js +1 -1
  207. package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
  208. package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
  209. package/dist/utils/EarlyTerminationSearch.js +164 -0
  210. package/dist/utils/ErrorHandler.d.ts +86 -0
  211. package/dist/utils/ErrorHandler.d.ts.map +1 -0
  212. package/dist/utils/ErrorHandler.js +201 -0
  213. package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
  214. package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
  215. package/dist/utils/FileDiscoveryUtil.js +169 -0
  216. package/dist/utils/GitHubRateLimiter.d.ts +88 -0
  217. package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
  218. package/dist/utils/GitHubRateLimiter.js +315 -0
  219. package/dist/utils/PerformanceMonitor.d.ts +134 -0
  220. package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
  221. package/dist/utils/PerformanceMonitor.js +347 -0
  222. package/dist/utils/RateLimiter.d.ts.map +1 -0
  223. package/dist/utils/RateLimiter.js +172 -0
  224. package/dist/utils/SecureDownloader.d.ts +241 -0
  225. package/dist/utils/SecureDownloader.d.ts.map +1 -0
  226. package/dist/utils/SecureDownloader.js +759 -0
  227. package/dist/utils/ToolCache.d.ts +82 -0
  228. package/dist/utils/ToolCache.d.ts.map +1 -0
  229. package/dist/utils/ToolCache.js +196 -0
  230. package/dist/utils/errorCodes.d.ts +136 -0
  231. package/dist/utils/errorCodes.d.ts.map +1 -0
  232. package/dist/utils/errorCodes.js +87 -0
  233. package/dist/utils/index.d.ts +3 -0
  234. package/dist/utils/index.d.ts.map +1 -1
  235. package/dist/utils/index.js +4 -1
  236. package/dist/utils/installation.d.ts +1 -1
  237. package/dist/utils/installation.d.ts.map +1 -1
  238. package/dist/utils/installation.js +9 -8
  239. package/dist/utils/searchUtils.d.ts +31 -0
  240. package/dist/utils/searchUtils.d.ts.map +1 -1
  241. package/dist/utils/searchUtils.js +62 -1
  242. package/package.json +17 -7
  243. package/dist/config/updateConfig.d.ts +0 -84
  244. package/dist/config/updateConfig.d.ts.map +0 -1
  245. package/dist/config/updateConfig.js +0 -148
  246. package/dist/server/tools/UpdateTools.d.ts +0 -10
  247. package/dist/server/tools/UpdateTools.d.ts.map +0 -1
  248. package/dist/server/tools/UpdateTools.js +0 -85
  249. package/dist/update/BackupManager.d.ts +0 -63
  250. package/dist/update/BackupManager.d.ts.map +0 -1
  251. package/dist/update/BackupManager.js +0 -370
  252. package/dist/update/DependencyChecker.d.ts +0 -41
  253. package/dist/update/DependencyChecker.d.ts.map +0 -1
  254. package/dist/update/DependencyChecker.js +0 -132
  255. package/dist/update/RateLimiter.d.ts.map +0 -1
  256. package/dist/update/RateLimiter.js +0 -172
  257. package/dist/update/SignatureVerifier.d.ts +0 -71
  258. package/dist/update/SignatureVerifier.d.ts.map +0 -1
  259. package/dist/update/SignatureVerifier.js +0 -214
  260. package/dist/update/UpdateChecker.d.ts +0 -132
  261. package/dist/update/UpdateChecker.d.ts.map +0 -1
  262. package/dist/update/UpdateChecker.js +0 -506
  263. package/dist/update/UpdateManager.d.ts +0 -60
  264. package/dist/update/UpdateManager.d.ts.map +0 -1
  265. package/dist/update/UpdateManager.js +0 -730
  266. package/dist/update/VersionManager.d.ts +0 -31
  267. package/dist/update/VersionManager.d.ts.map +0 -1
  268. package/dist/update/VersionManager.js +0 -181
  269. package/dist/update/index.d.ts +0 -9
  270. package/dist/update/index.d.ts.map +0 -1
  271. package/dist/update/index.js +0 -9
  272. /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
@@ -0,0 +1,759 @@
1
+ /**
2
+ * SecureDownloader - Reusable utility for safe content downloads
3
+ *
4
+ * Implements the validate-before-write pattern with comprehensive security features:
5
+ * - Content validation hooks (customizable validators)
6
+ * - Atomic file operations with temp files
7
+ * - Guaranteed cleanup on failure
8
+ * - Memory-efficient streaming for large files
9
+ * - Size limits to prevent DoS attacks
10
+ * - Path validation to prevent traversal
11
+ * - Timeout handling for network operations
12
+ * - Content type validation
13
+ *
14
+ * Usage Examples:
15
+ *
16
+ * // Basic download with validation
17
+ * const downloader = new SecureDownloader();
18
+ * await downloader.downloadToFile(
19
+ * 'https://example.com/file.md',
20
+ * './downloads/file.md',
21
+ * {
22
+ * validator: async (content) => ({
23
+ * isValid: !content.includes('malicious'),
24
+ * errorMessage: content.includes('malicious') ? 'Malicious content detected' : undefined
25
+ * }),
26
+ * maxSize: 1024 * 1024, // 1MB limit
27
+ * timeout: 30000 // 30 second timeout
28
+ * }
29
+ * );
30
+ *
31
+ * // Download to memory with validation
32
+ * const content = await downloader.downloadToMemory(
33
+ * 'https://example.com/data.json',
34
+ * {
35
+ * validator: async (content) => {
36
+ * try {
37
+ * JSON.parse(content);
38
+ * return { isValid: true };
39
+ * } catch {
40
+ * return { isValid: false, errorMessage: 'Invalid JSON format' };
41
+ * }
42
+ * }
43
+ * }
44
+ * );
45
+ *
46
+ * // Streaming download for large files
47
+ * await downloader.downloadStream(
48
+ * 'https://example.com/large-file.zip',
49
+ * './downloads/large-file.zip',
50
+ * {
51
+ * streamValidator: (chunk) => !chunk.includes(Buffer.from('VIRUS')),
52
+ * maxSize: 100 * 1024 * 1024, // 100MB limit
53
+ * timeout: 300000 // 5 minute timeout
54
+ * }
55
+ * );
56
+ */
57
+ import * as fs from 'fs/promises';
58
+ import * as path from 'path';
59
+ import { randomBytes, createHash } from 'crypto';
60
+ import { Readable } from 'stream';
61
+ import { pipeline } from 'stream/promises';
62
+ import { createWriteStream } from 'fs';
63
+ import { SecurityError } from '../errors/SecurityError.js';
64
+ import { SECURITY_LIMITS } from '../security/constants.js';
65
+ import { ContentValidator as SecurityContentValidator } from '../security/contentValidator.js';
66
+ import { PathValidator } from '../security/pathValidator.js';
67
+ import { FileLockManager } from '../security/fileLockManager.js';
68
+ import { SecurityMonitor } from '../security/securityMonitor.js';
69
+ import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
70
+ import { RateLimiter } from './RateLimiter.js';
71
+ import { logger } from './logger.js';
72
+ /**
73
+ * Custom error types for different failure scenarios
74
+ */
75
+ export class DownloadError extends Error {
76
+ code;
77
+ originalError;
78
+ constructor(message, code, originalError) {
79
+ super(message);
80
+ this.code = code;
81
+ this.originalError = originalError;
82
+ this.name = 'DownloadError';
83
+ }
84
+ static networkError(message, originalError) {
85
+ return new DownloadError(message, 'NETWORK_ERROR', originalError);
86
+ }
87
+ static validationError(message) {
88
+ return new DownloadError(message, 'VALIDATION_ERROR');
89
+ }
90
+ static securityError(message) {
91
+ return new DownloadError(message, 'SECURITY_ERROR');
92
+ }
93
+ static timeoutError(message) {
94
+ return new DownloadError(message, 'TIMEOUT_ERROR');
95
+ }
96
+ static filesystemError(message, originalError) {
97
+ return new DownloadError(message, 'FILESYSTEM_ERROR', originalError);
98
+ }
99
+ }
100
+ /**
101
+ * SecureDownloader - Implements validate-before-write pattern for safe downloads
102
+ *
103
+ * Key Security Features:
104
+ * 1. VALIDATE-BEFORE-WRITE: All content validation occurs before any disk operations
105
+ * 2. ATOMIC OPERATIONS: Uses temporary files with atomic rename to prevent corruption
106
+ * 3. GUARANTEED CLEANUP: Automatic cleanup of temporary files on any failure
107
+ * 4. SIZE LIMITS: Prevents DoS attacks through large file downloads
108
+ * 5. PATH VALIDATION: Prevents directory traversal attacks
109
+ * 6. TIMEOUT PROTECTION: Prevents hanging network operations
110
+ * 7. CONTENT VALIDATION: Extensible validation system for different content types
111
+ */
112
+ export class SecureDownloader {
113
+ defaultTimeout;
114
+ defaultMaxSize;
115
+ tempDir;
116
+ globalRateLimiter;
117
+ urlRateLimiters;
118
+ constructor(options) {
119
+ this.defaultTimeout = options?.defaultTimeout || 30000; // 30 seconds
120
+ this.defaultMaxSize = options?.defaultMaxSize || SECURITY_LIMITS.MAX_FILE_SIZE;
121
+ this.tempDir = options?.tempDir || '.tmp';
122
+ // Initialize rate limiters
123
+ const rateLimitConfig = options?.rateLimitOptions || {};
124
+ this.globalRateLimiter = new RateLimiter({
125
+ maxRequests: rateLimitConfig.maxGlobalRequests || 100, // 100 downloads per hour globally
126
+ windowMs: rateLimitConfig.windowMs || 60 * 60 * 1000, // 1 hour
127
+ minDelayMs: 1000 // Minimum 1 second between requests
128
+ });
129
+ this.urlRateLimiters = new Map();
130
+ }
131
+ /**
132
+ * Download content to a file with validation
133
+ *
134
+ * SECURITY: Implements validate-before-write pattern:
135
+ * 1. Download content to memory
136
+ * 2. Validate all content
137
+ * 3. Only then write to disk atomically
138
+ *
139
+ * @param url - URL to download from
140
+ * @param destinationPath - Local file path to save to
141
+ * @param options - Download and validation options
142
+ */
143
+ async downloadToFile(url, destinationPath, options = {}) {
144
+ const startTime = Date.now();
145
+ logger.debug(`Starting secure download from ${url} to ${destinationPath}`);
146
+ try {
147
+ // SECURITY: Validate URL and destination path first
148
+ this.validateUrl(url);
149
+ const validatedPath = await this.validateDestinationPath(destinationPath);
150
+ // SECURITY: Check if file already exists (prevent accidental overwrites)
151
+ try {
152
+ await fs.access(validatedPath);
153
+ throw DownloadError.filesystemError(`File already exists: ${destinationPath}`);
154
+ }
155
+ catch (error) {
156
+ if (error.code !== 'ENOENT') {
157
+ throw error; // Re-throw if it's not a "file not found" error
158
+ }
159
+ // File doesn't exist, proceed with download
160
+ }
161
+ // STEP 1: Check rate limits before download
162
+ await this.checkRateLimit(url);
163
+ // STEP 2: Download content to memory (no disk operations yet)
164
+ const content = await this.downloadToMemory(url, options);
165
+ // STEP 3: Validate checksum if provided
166
+ if (options.expectedChecksum) {
167
+ await this.validateChecksum(content, options.expectedChecksum);
168
+ }
169
+ // STEP 4: All validation is complete, now write atomically
170
+ const useAtomic = options.atomic !== false; // Default to true
171
+ if (useAtomic) {
172
+ await this.atomicWriteFile(validatedPath, content);
173
+ }
174
+ else {
175
+ await this.directWriteFile(validatedPath, content);
176
+ }
177
+ const duration = Date.now() - startTime;
178
+ logger.info(`Secure download completed: ${destinationPath} (${content.length} bytes, ${duration}ms)`);
179
+ // Log successful download for security monitoring
180
+ SecurityMonitor.logSecurityEvent({
181
+ type: 'FILE_COPIED',
182
+ severity: 'LOW',
183
+ source: 'secure_downloader',
184
+ details: `Downloaded ${content.length} bytes from ${url} to ${destinationPath}`,
185
+ metadata: {
186
+ url,
187
+ destinationPath,
188
+ contentLength: content.length,
189
+ duration
190
+ }
191
+ });
192
+ }
193
+ catch (error) {
194
+ const duration = Date.now() - startTime;
195
+ logger.error(`Secure download failed: ${error instanceof Error ? error.message : String(error)}`);
196
+ // Log failed download for security monitoring
197
+ SecurityMonitor.logSecurityEvent({
198
+ type: 'PATH_TRAVERSAL_ATTEMPT',
199
+ severity: 'MEDIUM',
200
+ source: 'secure_downloader',
201
+ details: `Download failed: ${error instanceof Error ? error.message : String(error)}`,
202
+ metadata: {
203
+ url,
204
+ destinationPath,
205
+ duration,
206
+ errorType: error instanceof DownloadError ? error.code : 'UNKNOWN'
207
+ }
208
+ });
209
+ throw error;
210
+ }
211
+ }
212
+ /**
213
+ * Download content to memory with validation
214
+ *
215
+ * @param url - URL to download from
216
+ * @param options - Download and validation options
217
+ * @returns Validated content as string
218
+ */
219
+ async downloadToMemory(url, options = {}) {
220
+ const timeout = options.timeout || this.defaultTimeout;
221
+ const maxSize = options.maxSize || this.defaultMaxSize;
222
+ logger.debug(`Downloading content from ${url} (max: ${maxSize} bytes, timeout: ${timeout}ms)`);
223
+ try {
224
+ // SECURITY: Validate URL format
225
+ this.validateUrl(url);
226
+ // STEP 1: Check rate limits before download
227
+ await this.checkRateLimit(url);
228
+ // STEP 2: Fetch content with size and timeout protection
229
+ const content = await this.fetchWithLimits(url, maxSize, timeout, options.headers);
230
+ // STEP 3: Validate content type if specified
231
+ if (options.expectedContentType) {
232
+ await this.validateContentType(content, options.expectedContentType);
233
+ }
234
+ // STEP 4: Validate checksum if provided
235
+ if (options.expectedChecksum) {
236
+ await this.validateChecksum(content, options.expectedChecksum);
237
+ }
238
+ // STEP 5: Run built-in security validation
239
+ const securityResult = SecurityContentValidator.validateAndSanitize(content);
240
+ if (!securityResult.isValid && securityResult.severity === 'critical') {
241
+ throw DownloadError.securityError(`Critical security threat detected: ${securityResult.detectedPatterns?.join(', ')}`);
242
+ }
243
+ // STEP 6: Run custom validator if provided
244
+ if (options.validator) {
245
+ logger.debug('Running custom content validation');
246
+ const validationResult = await options.validator(content);
247
+ if (!validationResult.isValid) {
248
+ throw DownloadError.validationError(validationResult.errorMessage || 'Content validation failed');
249
+ }
250
+ }
251
+ logger.debug(`Content validation passed (${content.length} bytes)`);
252
+ return securityResult.sanitizedContent || content;
253
+ }
254
+ catch (error) {
255
+ if (error instanceof DownloadError) {
256
+ throw error;
257
+ }
258
+ throw DownloadError.networkError(`Failed to download content from ${url}: ${error instanceof Error ? error.message : String(error)}`, error instanceof Error ? error : undefined);
259
+ }
260
+ }
261
+ /**
262
+ * Download large files using streaming with chunk-level validation
263
+ *
264
+ * @param url - URL to download from
265
+ * @param destinationPath - Local file path to save to
266
+ * @param options - Streaming download options
267
+ */
268
+ async downloadStream(url, destinationPath, options = {}) {
269
+ const startTime = Date.now();
270
+ const maxSize = options.maxSize || this.defaultMaxSize;
271
+ const timeout = options.timeout || this.defaultTimeout;
272
+ logger.debug(`Starting streaming download from ${url} to ${destinationPath}`);
273
+ try {
274
+ // SECURITY: Check rate limits before download
275
+ await this.checkRateLimit(url);
276
+ // SECURITY: Validate URL and destination path
277
+ this.validateUrl(url);
278
+ const validatedPath = await this.validateDestinationPath(destinationPath);
279
+ // Generate temporary file path for atomic operation
280
+ const tempPath = await this.getTempFilePath(validatedPath);
281
+ let downloadedSize = 0;
282
+ let timeoutHandle;
283
+ // Create abort controller for timeout handling
284
+ const abortController = new AbortController();
285
+ timeoutHandle = setTimeout(() => {
286
+ abortController.abort();
287
+ }, timeout);
288
+ try {
289
+ // SECURITY: Fetch with abort signal for timeout
290
+ const response = await fetch(url, {
291
+ signal: abortController.signal,
292
+ headers: options.headers
293
+ });
294
+ if (!response.ok) {
295
+ throw new Error(`HTTP ${response.status}: ${response.statusText}`);
296
+ }
297
+ if (!response.body) {
298
+ throw new Error('Response body is null');
299
+ }
300
+ // Ensure temp directory exists
301
+ await fs.mkdir(path.dirname(tempPath), { recursive: true });
302
+ // Create write stream to temporary file
303
+ const writeStream = createWriteStream(tempPath);
304
+ // Create a transform stream for validation and size checking
305
+ const validationStream = new Readable({
306
+ async read() {
307
+ // This stream will be fed by the pipeline
308
+ }
309
+ });
310
+ // Set up chunk validation and size checking
311
+ const reader = response.body.getReader();
312
+ const pump = async () => {
313
+ try {
314
+ while (true) {
315
+ const { done, value } = await reader.read();
316
+ if (done)
317
+ break;
318
+ // SECURITY: Check size limit
319
+ downloadedSize += value.length;
320
+ if (downloadedSize > maxSize) {
321
+ throw DownloadError.securityError(`File size exceeds limit: ${downloadedSize} > ${maxSize} bytes`);
322
+ }
323
+ // SECURITY: Run chunk validator if provided
324
+ if (options.streamValidator && !options.streamValidator(value)) {
325
+ throw DownloadError.validationError('Chunk validation failed');
326
+ }
327
+ validationStream.push(value);
328
+ }
329
+ validationStream.push(null); // End stream
330
+ }
331
+ catch (error) {
332
+ validationStream.destroy(error instanceof Error ? error : new Error(String(error)));
333
+ }
334
+ };
335
+ // Start the pump and pipeline concurrently
336
+ const [pumpResult] = await Promise.all([
337
+ pump(),
338
+ pipeline(validationStream, writeStream)
339
+ ]);
340
+ // Clear timeout
341
+ if (timeoutHandle) {
342
+ clearTimeout(timeoutHandle);
343
+ timeoutHandle = undefined;
344
+ }
345
+ // SECURITY: Atomic rename to final destination
346
+ await fs.rename(tempPath, validatedPath);
347
+ const duration = Date.now() - startTime;
348
+ logger.info(`Streaming download completed: ${destinationPath} (${downloadedSize} bytes, ${duration}ms)`);
349
+ // Log successful streaming download
350
+ SecurityMonitor.logSecurityEvent({
351
+ type: 'FILE_COPIED',
352
+ severity: 'LOW',
353
+ source: 'secure_downloader',
354
+ details: `Streamed ${downloadedSize} bytes from ${url} to ${destinationPath}`,
355
+ metadata: {
356
+ url,
357
+ destinationPath,
358
+ contentLength: downloadedSize,
359
+ duration
360
+ }
361
+ });
362
+ }
363
+ catch (error) {
364
+ // SECURITY: Guaranteed cleanup of temporary file
365
+ try {
366
+ await fs.unlink(tempPath);
367
+ logger.debug(`Cleaned up temp file: ${tempPath}`);
368
+ }
369
+ catch (cleanupError) {
370
+ logger.warn(`Failed to clean up temp file ${tempPath}: ${cleanupError}`);
371
+ }
372
+ throw error;
373
+ }
374
+ finally {
375
+ if (timeoutHandle) {
376
+ clearTimeout(timeoutHandle);
377
+ }
378
+ }
379
+ }
380
+ catch (error) {
381
+ const duration = Date.now() - startTime;
382
+ logger.error(`Streaming download failed: ${error instanceof Error ? error.message : String(error)}`);
383
+ // Log failed streaming download
384
+ SecurityMonitor.logSecurityEvent({
385
+ type: 'PATH_TRAVERSAL_ATTEMPT',
386
+ severity: 'MEDIUM',
387
+ source: 'secure_downloader',
388
+ details: `Streaming download failed: ${error instanceof Error ? error.message : String(error)}`,
389
+ metadata: {
390
+ url,
391
+ destinationPath,
392
+ duration,
393
+ errorType: error instanceof DownloadError ? error.code : 'UNKNOWN'
394
+ }
395
+ });
396
+ if (error instanceof Error && error.name === 'AbortError') {
397
+ throw DownloadError.timeoutError(`Download timed out after ${timeout}ms`);
398
+ }
399
+ if (error instanceof DownloadError) {
400
+ throw error;
401
+ }
402
+ throw DownloadError.networkError(`Streaming download failed: ${error instanceof Error ? error.message : String(error)}`, error instanceof Error ? error : undefined);
403
+ }
404
+ }
405
+ /**
406
+ * Validate URL format and security with Unicode normalization
407
+ */
408
+ validateUrl(url) {
409
+ if (!url || typeof url !== 'string') {
410
+ throw DownloadError.validationError('URL must be a non-empty string');
411
+ }
412
+ // SECURITY FIX: DMCP-SEC-004 - Unicode normalization on user input
413
+ const unicodeValidation = UnicodeValidator.normalize(url);
414
+ const normalizedUrl = unicodeValidation.normalizedContent;
415
+ if (!unicodeValidation.isValid) {
416
+ SecurityMonitor.logSecurityEvent({
417
+ type: 'UNICODE_VALIDATION_ERROR',
418
+ severity: 'MEDIUM',
419
+ source: 'secure_downloader',
420
+ details: `URL contains suspicious Unicode patterns: ${unicodeValidation.detectedIssues?.join(', ')}`,
421
+ metadata: { originalUrl: url, normalizedUrl }
422
+ });
423
+ }
424
+ // Use normalized URL for further validation
425
+ url = normalizedUrl;
426
+ let parsedUrl;
427
+ try {
428
+ parsedUrl = new URL(url);
429
+ }
430
+ catch (error) {
431
+ throw DownloadError.validationError(`Invalid URL format: ${url}`);
432
+ }
433
+ // SECURITY: Only allow HTTPS and HTTP protocols
434
+ if (!['https:', 'http:'].includes(parsedUrl.protocol)) {
435
+ throw DownloadError.securityError(`Unsupported protocol: ${parsedUrl.protocol}. Only HTTP/HTTPS allowed.`);
436
+ }
437
+ // SECURITY: Prevent requests to localhost/private networks
438
+ const hostname = parsedUrl.hostname.toLowerCase();
439
+ if (hostname === 'localhost' || hostname === '127.0.0.1' || hostname === '::1') {
440
+ throw DownloadError.securityError('Downloads from localhost are not allowed');
441
+ }
442
+ // SECURITY: Check for private IP ranges (basic protection)
443
+ if (hostname.startsWith('192.168.') || hostname.startsWith('10.') || hostname.startsWith('172.')) {
444
+ throw DownloadError.securityError('Downloads from private IP ranges are not allowed');
445
+ }
446
+ }
447
+ /**
448
+ * Validate destination path for security
449
+ */
450
+ async validateDestinationPath(filePath) {
451
+ try {
452
+ // Use existing PathValidator for comprehensive path validation
453
+ return await PathValidator.validatePersonaPath(filePath);
454
+ }
455
+ catch (error) {
456
+ throw DownloadError.securityError(`Invalid destination path: ${error instanceof Error ? error.message : String(error)}`);
457
+ }
458
+ }
459
+ /**
460
+ * Fetch content with size and timeout limits
461
+ */
462
+ async fetchWithLimits(url, maxSize, timeout, headers) {
463
+ const abortController = new AbortController();
464
+ const timeoutHandle = setTimeout(() => abortController.abort(), timeout);
465
+ try {
466
+ const response = await fetch(url, {
467
+ signal: abortController.signal,
468
+ headers: {
469
+ 'User-Agent': 'DollhouseMCP-SecureDownloader/1.0',
470
+ ...headers
471
+ }
472
+ });
473
+ if (!response.ok) {
474
+ throw new Error(`HTTP ${response.status}: ${response.statusText}`);
475
+ }
476
+ // SECURITY: Check Content-Length header if available
477
+ const contentLength = response.headers.get('content-length');
478
+ if (contentLength && parseInt(contentLength, 10) > maxSize) {
479
+ throw DownloadError.securityError(`Content size ${contentLength} exceeds limit of ${maxSize} bytes`);
480
+ }
481
+ // Read content with size checking
482
+ const chunks = [];
483
+ let totalSize = 0;
484
+ if (!response.body) {
485
+ throw new Error('Response body is null');
486
+ }
487
+ const reader = response.body.getReader();
488
+ try {
489
+ while (true) {
490
+ const { done, value } = await reader.read();
491
+ if (done)
492
+ break;
493
+ totalSize += value.length;
494
+ if (totalSize > maxSize) {
495
+ throw DownloadError.securityError(`Content size ${totalSize} exceeds limit of ${maxSize} bytes`);
496
+ }
497
+ chunks.push(value);
498
+ }
499
+ }
500
+ finally {
501
+ reader.releaseLock();
502
+ }
503
+ // Combine chunks and decode
504
+ const totalLength = chunks.reduce((sum, chunk) => sum + chunk.length, 0);
505
+ const combined = new Uint8Array(totalLength);
506
+ let offset = 0;
507
+ for (const chunk of chunks) {
508
+ combined.set(chunk, offset);
509
+ offset += chunk.length;
510
+ }
511
+ return new TextDecoder('utf-8').decode(combined);
512
+ }
513
+ catch (error) {
514
+ if (error instanceof Error && error.name === 'AbortError') {
515
+ throw DownloadError.timeoutError(`Request timed out after ${timeout}ms`);
516
+ }
517
+ throw error;
518
+ }
519
+ finally {
520
+ clearTimeout(timeoutHandle);
521
+ }
522
+ }
523
+ /**
524
+ * Validate content type if specified
525
+ */
526
+ async validateContentType(content, expectedType) {
527
+ // Basic content type validation based on content analysis
528
+ switch (expectedType.toLowerCase()) {
529
+ case 'json':
530
+ try {
531
+ JSON.parse(content);
532
+ }
533
+ catch {
534
+ throw DownloadError.validationError('Content is not valid JSON');
535
+ }
536
+ break;
537
+ case 'yaml':
538
+ case 'yml':
539
+ // Use existing YAML validation
540
+ if (!SecurityContentValidator.validateYamlContent(content)) {
541
+ throw DownloadError.validationError('Content is not valid YAML');
542
+ }
543
+ break;
544
+ case 'markdown':
545
+ case 'md':
546
+ // Basic markdown validation (check for frontmatter format)
547
+ if (content.startsWith('---')) {
548
+ const frontmatterEnd = content.indexOf('\n---\n', 3);
549
+ if (frontmatterEnd === -1) {
550
+ throw DownloadError.validationError('Invalid markdown frontmatter format');
551
+ }
552
+ }
553
+ break;
554
+ default:
555
+ logger.debug(`No specific validation for content type: ${expectedType}`);
556
+ }
557
+ }
558
+ /**
559
+ * Atomic file write using FileLockManager
560
+ */
561
+ async atomicWriteFile(filePath, content) {
562
+ const resource = `download:${filePath}`;
563
+ await FileLockManager.withLock(resource, async () => {
564
+ // Ensure directory exists
565
+ await fs.mkdir(path.dirname(filePath), { recursive: true });
566
+ // Use FileLockManager's atomic write
567
+ await FileLockManager.atomicWriteFile(filePath, content, { encoding: 'utf-8' });
568
+ });
569
+ }
570
+ /**
571
+ * Direct file write (non-atomic, for when atomic is disabled)
572
+ */
573
+ async directWriteFile(filePath, content) {
574
+ // Ensure directory exists
575
+ await fs.mkdir(path.dirname(filePath), { recursive: true });
576
+ // Direct write
577
+ await fs.writeFile(filePath, content, 'utf-8');
578
+ }
579
+ /**
580
+ * Generate temporary file path for atomic operations
581
+ */
582
+ async getTempFilePath(originalPath) {
583
+ const dir = path.dirname(originalPath);
584
+ const basename = path.basename(originalPath);
585
+ const random = randomBytes(8).toString('hex');
586
+ const tempDir = path.join(dir, this.tempDir);
587
+ // Ensure temp directory exists
588
+ await fs.mkdir(tempDir, { recursive: true });
589
+ return path.join(tempDir, `${basename}.${random}.tmp`);
590
+ }
591
+ /**
592
+ * Check rate limits for downloads
593
+ */
594
+ async checkRateLimit(url) {
595
+ // Check global rate limit
596
+ const globalStatus = this.globalRateLimiter.checkLimit();
597
+ if (!globalStatus.allowed) {
598
+ SecurityMonitor.logSecurityEvent({
599
+ type: 'RATE_LIMIT_EXCEEDED',
600
+ severity: 'MEDIUM',
601
+ source: 'secure_downloader',
602
+ details: `Global download rate limit exceeded. Retry after ${globalStatus.retryAfterMs}ms`,
603
+ metadata: { url, retryAfterMs: globalStatus.retryAfterMs }
604
+ });
605
+ throw DownloadError.securityError(`Download rate limit exceeded. Please retry after ${Math.ceil(globalStatus.retryAfterMs / 1000)} seconds`);
606
+ }
607
+ // Check per-URL rate limit
608
+ const parsedUrl = new URL(url);
609
+ const urlKey = `${parsedUrl.hostname}:${parsedUrl.port || (parsedUrl.protocol === 'https:' ? '443' : '80')}`;
610
+ if (!this.urlRateLimiters.has(urlKey)) {
611
+ this.urlRateLimiters.set(urlKey, new RateLimiter({
612
+ maxRequests: 10, // 10 requests per hour per URL
613
+ windowMs: 60 * 60 * 1000,
614
+ minDelayMs: 5000 // 5 second minimum delay between requests to same URL
615
+ }));
616
+ }
617
+ const urlLimiter = this.urlRateLimiters.get(urlKey);
618
+ const urlStatus = urlLimiter.checkLimit();
619
+ if (!urlStatus.allowed) {
620
+ SecurityMonitor.logSecurityEvent({
621
+ type: 'RATE_LIMIT_EXCEEDED',
622
+ severity: 'MEDIUM',
623
+ source: 'secure_downloader',
624
+ details: `Per-URL download rate limit exceeded for ${urlKey}. Retry after ${urlStatus.retryAfterMs}ms`,
625
+ metadata: { url, urlKey, retryAfterMs: urlStatus.retryAfterMs }
626
+ });
627
+ throw DownloadError.securityError(`Too many requests to ${urlKey}. Please retry after ${Math.ceil(urlStatus.retryAfterMs / 1000)} seconds`);
628
+ }
629
+ // Consume rate limit tokens
630
+ this.globalRateLimiter.consumeToken();
631
+ urlLimiter.consumeToken();
632
+ }
633
+ /**
634
+ * Validate content checksum for integrity verification
635
+ */
636
+ async validateChecksum(content, expectedChecksum) {
637
+ const normalizedExpected = expectedChecksum.toLowerCase().trim();
638
+ // Validate checksum format (SHA-256 should be 64 hex characters)
639
+ if (!/^[a-f0-9]{64}$/.test(normalizedExpected)) {
640
+ throw DownloadError.validationError('Invalid checksum format. Expected SHA-256 (64 hex characters)');
641
+ }
642
+ const contentBuffer = Buffer.from(content, 'utf-8');
643
+ const actualChecksum = createHash('sha256').update(contentBuffer).digest('hex');
644
+ if (actualChecksum !== normalizedExpected) {
645
+ SecurityMonitor.logSecurityEvent({
646
+ type: 'CONTENT_INJECTION_ATTEMPT',
647
+ severity: 'HIGH',
648
+ source: 'secure_downloader',
649
+ details: `Checksum mismatch detected - possible content tampering`,
650
+ metadata: {
651
+ expectedChecksum: normalizedExpected,
652
+ actualChecksum,
653
+ contentLength: content.length
654
+ }
655
+ });
656
+ throw DownloadError.securityError(`Content checksum verification failed. Expected: ${normalizedExpected}, Got: ${actualChecksum}`);
657
+ }
658
+ logger.debug(`Checksum validation passed: ${actualChecksum}`);
659
+ }
660
+ /**
661
+ * Create a content validator that combines multiple validators
662
+ */
663
+ static combineValidators(...validators) {
664
+ return async (content) => {
665
+ for (const validator of validators) {
666
+ const result = await validator(content);
667
+ if (!result.isValid) {
668
+ return result;
669
+ }
670
+ }
671
+ return { isValid: true };
672
+ };
673
+ }
674
+ /**
675
+ * Create a content validator for JSON content
676
+ */
677
+ static jsonValidator() {
678
+ return async (content) => {
679
+ try {
680
+ JSON.parse(content);
681
+ return { isValid: true };
682
+ }
683
+ catch (error) {
684
+ return {
685
+ isValid: false,
686
+ errorMessage: `Invalid JSON: ${error instanceof Error ? error.message : String(error)}`,
687
+ severity: 'medium'
688
+ };
689
+ }
690
+ };
691
+ }
692
+ /**
693
+ * Create a content validator for YAML content
694
+ */
695
+ static yamlValidator() {
696
+ return async (content) => {
697
+ const isValid = SecurityContentValidator.validateYamlContent(content);
698
+ return {
699
+ isValid,
700
+ errorMessage: isValid ? undefined : 'Invalid or malicious YAML content',
701
+ severity: isValid ? 'low' : 'high'
702
+ };
703
+ };
704
+ }
705
+ /**
706
+ * Create a content validator for markdown content
707
+ */
708
+ static markdownValidator() {
709
+ return async (content) => {
710
+ try {
711
+ // Use existing persona content sanitization for markdown
712
+ SecurityContentValidator.sanitizePersonaContent(content);
713
+ return { isValid: true };
714
+ }
715
+ catch (error) {
716
+ return {
717
+ isValid: false,
718
+ errorMessage: `Invalid markdown: ${error instanceof Error ? error.message : String(error)}`,
719
+ severity: error instanceof SecurityError ? 'critical' : 'medium'
720
+ };
721
+ }
722
+ };
723
+ }
724
+ /**
725
+ * Create a content validator with size limits
726
+ */
727
+ static sizeValidator(maxSize) {
728
+ return async (content) => {
729
+ const size = Buffer.byteLength(content, 'utf-8');
730
+ if (size > maxSize) {
731
+ return {
732
+ isValid: false,
733
+ errorMessage: `Content size ${size} exceeds limit of ${maxSize} bytes`,
734
+ severity: 'high'
735
+ };
736
+ }
737
+ return { isValid: true };
738
+ };
739
+ }
740
+ /**
741
+ * Create a content validator that checks for forbidden patterns
742
+ */
743
+ static patternValidator(forbiddenPatterns, errorMessage = 'Forbidden pattern detected') {
744
+ return async (content) => {
745
+ for (const pattern of forbiddenPatterns) {
746
+ if (pattern.test(content)) {
747
+ return {
748
+ isValid: false,
749
+ errorMessage,
750
+ severity: 'high',
751
+ metadata: { pattern: pattern.source }
752
+ };
753
+ }
754
+ }
755
+ return { isValid: true };
756
+ };
757
+ }
758
+ }
759
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SecureDownloader.js","sourceRoot":"","sources":["../../src/utils/SecureDownloader.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * SecureDownloader - Reusable utility for safe content downloads\n * \n * Implements the validate-before-write pattern with comprehensive security features:\n * - Content validation hooks (customizable validators)\n * - Atomic file operations with temp files\n * - Guaranteed cleanup on failure\n * - Memory-efficient streaming for large files\n * - Size limits to prevent DoS attacks\n * - Path validation to prevent traversal\n * - Timeout handling for network operations\n * - Content type validation\n * \n * Usage Examples:\n * \n * // Basic download with validation\n * const downloader = new SecureDownloader();\n * await downloader.downloadToFile(\n *   'https://example.com/file.md',\n *   './downloads/file.md',\n *   {\n *     validator: async (content) => ({\n *       isValid: !content.includes('malicious'),\n *       errorMessage: content.includes('malicious') ? 'Malicious content detected' : undefined\n *     }),\n *     maxSize: 1024 * 1024, // 1MB limit\n *     timeout: 30000 // 30 second timeout\n *   }\n * );\n * \n * // Download to memory with validation\n * const content = await downloader.downloadToMemory(\n *   'https://example.com/data.json',\n *   {\n *     validator: async (content) => {\n *       try {\n *         JSON.parse(content);\n *         return { isValid: true };\n *       } catch {\n *         return { isValid: false, errorMessage: 'Invalid JSON format' };\n *       }\n *     }\n *   }\n * );\n * \n * // Streaming download for large files\n * await downloader.downloadStream(\n *   'https://example.com/large-file.zip',\n *   './downloads/large-file.zip',\n *   {\n *     streamValidator: (chunk) => !chunk.includes(Buffer.from('VIRUS')),\n *     maxSize: 100 * 1024 * 1024, // 100MB limit\n *     timeout: 300000 // 5 minute timeout\n *   }\n * );\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { randomBytes, createHash } from 'crypto';\nimport { Readable } from 'stream';\nimport { pipeline } from 'stream/promises';\nimport { createWriteStream } from 'fs';\n\nimport { SecurityError } from '../errors/SecurityError.js';\nimport { SECURITY_LIMITS } from '../security/constants.js';\nimport { ContentValidator as SecurityContentValidator } from '../security/contentValidator.js';\nimport { PathValidator } from '../security/pathValidator.js';\nimport { FileLockManager } from '../security/fileLockManager.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { RateLimiter } from './RateLimiter.js';\nimport { logger } from './logger.js';\n\n/**\n * Result of content validation\n */\nexport interface ValidationResult {\n  /** Whether the content is valid and safe */\n  isValid: boolean;\n  /** Error message if validation failed */\n  errorMessage?: string;\n  /** Severity of any detected issues */\n  severity?: 'low' | 'medium' | 'high' | 'critical';\n  /** Additional metadata about validation */\n  metadata?: Record<string, any>;\n}\n\n/**\n * Content validator function type\n */\nexport type ContentValidatorFunction = (content: string) => Promise<ValidationResult>;\n\n/**\n * Stream chunk validator function type\n */\nexport type StreamValidator = (chunk: Uint8Array) => boolean;\n\n/**\n * Options for download operations\n */\nexport interface DownloadOptions {\n  /** Custom content validator function */\n  validator?: ContentValidatorFunction;\n  /** Maximum file size in bytes (default: SECURITY_LIMITS.MAX_FILE_SIZE) */\n  maxSize?: number;\n  /** Network timeout in milliseconds (default: 30000) */\n  timeout?: number;\n  /** Whether to use atomic file operations (default: true) */\n  atomic?: boolean;\n  /** Expected content type (for validation) */\n  expectedContentType?: string;\n  /** Custom HTTP headers */\n  headers?: Record<string, string>;\n  /** Expected SHA-256 checksum for integrity validation */\n  expectedChecksum?: string;\n}\n\n/**\n * Options for streaming downloads\n */\nexport interface StreamDownloadOptions {\n  /** Chunk-level validator for streaming validation */\n  streamValidator?: StreamValidator;\n  /** Maximum file size in bytes (default: SECURITY_LIMITS.MAX_FILE_SIZE) */\n  maxSize?: number;\n  /** Network timeout in milliseconds (default: 30000) */\n  timeout?: number;\n  /** Custom HTTP headers */\n  headers?: Record<string, string>;\n}\n\n/**\n * Custom error types for different failure scenarios\n */\nexport class DownloadError extends Error {\n  constructor(\n    message: string,\n    public readonly code: string,\n    public readonly originalError?: Error\n  ) {\n    super(message);\n    this.name = 'DownloadError';\n  }\n\n  static networkError(message: string, originalError?: Error): DownloadError {\n    return new DownloadError(message, 'NETWORK_ERROR', originalError);\n  }\n\n  static validationError(message: string): DownloadError {\n    return new DownloadError(message, 'VALIDATION_ERROR');\n  }\n\n  static securityError(message: string): DownloadError {\n    return new DownloadError(message, 'SECURITY_ERROR');\n  }\n\n  static timeoutError(message: string): DownloadError {\n    return new DownloadError(message, 'TIMEOUT_ERROR');\n  }\n\n  static filesystemError(message: string, originalError?: Error): DownloadError {\n    return new DownloadError(message, 'FILESYSTEM_ERROR', originalError);\n  }\n}\n\n/**\n * SecureDownloader - Implements validate-before-write pattern for safe downloads\n * \n * Key Security Features:\n * 1. VALIDATE-BEFORE-WRITE: All content validation occurs before any disk operations\n * 2. ATOMIC OPERATIONS: Uses temporary files with atomic rename to prevent corruption\n * 3. GUARANTEED CLEANUP: Automatic cleanup of temporary files on any failure\n * 4. SIZE LIMITS: Prevents DoS attacks through large file downloads\n * 5. PATH VALIDATION: Prevents directory traversal attacks\n * 6. TIMEOUT PROTECTION: Prevents hanging network operations\n * 7. CONTENT VALIDATION: Extensible validation system for different content types\n */\nexport class SecureDownloader {\n  private readonly defaultTimeout: number;\n  private readonly defaultMaxSize: number;\n  private readonly tempDir: string;\n  private readonly globalRateLimiter: RateLimiter;\n  private readonly urlRateLimiters: Map<string, RateLimiter>;\n\n  constructor(options?: {\n    defaultTimeout?: number;\n    defaultMaxSize?: number;\n    tempDir?: string;\n    rateLimitOptions?: {\n      maxRequestsPerUrl?: number;\n      maxGlobalRequests?: number;\n      windowMs?: number;\n    };\n  }) {\n    this.defaultTimeout = options?.defaultTimeout || 30000; // 30 seconds\n    this.defaultMaxSize = options?.defaultMaxSize || SECURITY_LIMITS.MAX_FILE_SIZE;\n    this.tempDir = options?.tempDir || '.tmp';\n    \n    // Initialize rate limiters\n    const rateLimitConfig = options?.rateLimitOptions || {};\n    this.globalRateLimiter = new RateLimiter({\n      maxRequests: rateLimitConfig.maxGlobalRequests || 100, // 100 downloads per hour globally\n      windowMs: rateLimitConfig.windowMs || 60 * 60 * 1000, // 1 hour\n      minDelayMs: 1000 // Minimum 1 second between requests\n    });\n    this.urlRateLimiters = new Map();\n  }\n\n  /**\n   * Download content to a file with validation\n   * \n   * SECURITY: Implements validate-before-write pattern:\n   * 1. Download content to memory\n   * 2. Validate all content\n   * 3. Only then write to disk atomically\n   * \n   * @param url - URL to download from\n   * @param destinationPath - Local file path to save to\n   * @param options - Download and validation options\n   */\n  async downloadToFile(\n    url: string,\n    destinationPath: string,\n    options: DownloadOptions = {}\n  ): Promise<void> {\n    const startTime = Date.now();\n    logger.debug(`Starting secure download from ${url} to ${destinationPath}`);\n\n    try {\n      // SECURITY: Validate URL and destination path first\n      this.validateUrl(url);\n      const validatedPath = await this.validateDestinationPath(destinationPath);\n\n      // SECURITY: Check if file already exists (prevent accidental overwrites)\n      try {\n        await fs.access(validatedPath);\n        throw DownloadError.filesystemError(`File already exists: ${destinationPath}`);\n      } catch (error) {\n        if ((error as NodeJS.ErrnoException).code !== 'ENOENT') {\n          throw error; // Re-throw if it's not a \"file not found\" error\n        }\n        // File doesn't exist, proceed with download\n      }\n\n      // STEP 1: Check rate limits before download\n      await this.checkRateLimit(url);\n\n      // STEP 2: Download content to memory (no disk operations yet)\n      const content = await this.downloadToMemory(url, options);\n\n      // STEP 3: Validate checksum if provided\n      if (options.expectedChecksum) {\n        await this.validateChecksum(content, options.expectedChecksum);\n      }\n\n      // STEP 4: All validation is complete, now write atomically\n      const useAtomic = options.atomic !== false; // Default to true\n      if (useAtomic) {\n        await this.atomicWriteFile(validatedPath, content);\n      } else {\n        await this.directWriteFile(validatedPath, content);\n      }\n\n      const duration = Date.now() - startTime;\n      logger.info(`Secure download completed: ${destinationPath} (${content.length} bytes, ${duration}ms)`);\n\n      // Log successful download for security monitoring\n      SecurityMonitor.logSecurityEvent({\n        type: 'FILE_COPIED',\n        severity: 'LOW',\n        source: 'secure_downloader',\n        details: `Downloaded ${content.length} bytes from ${url} to ${destinationPath}`,\n        metadata: {\n          url,\n          destinationPath,\n          contentLength: content.length,\n          duration\n        }\n      });\n\n    } catch (error) {\n      const duration = Date.now() - startTime;\n      logger.error(`Secure download failed: ${error instanceof Error ? error.message : String(error)}`);\n\n      // Log failed download for security monitoring\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'MEDIUM',\n        source: 'secure_downloader',\n        details: `Download failed: ${error instanceof Error ? error.message : String(error)}`,\n        metadata: {\n          url,\n          destinationPath,\n          duration,\n          errorType: error instanceof DownloadError ? error.code : 'UNKNOWN'\n        }\n      });\n\n      throw error;\n    }\n  }\n\n  /**\n   * Download content to memory with validation\n   * \n   * @param url - URL to download from\n   * @param options - Download and validation options\n   * @returns Validated content as string\n   */\n  async downloadToMemory(\n    url: string,\n    options: DownloadOptions = {}\n  ): Promise<string> {\n    const timeout = options.timeout || this.defaultTimeout;\n    const maxSize = options.maxSize || this.defaultMaxSize;\n\n    logger.debug(`Downloading content from ${url} (max: ${maxSize} bytes, timeout: ${timeout}ms)`);\n\n    try {\n      // SECURITY: Validate URL format\n      this.validateUrl(url);\n\n      // STEP 1: Check rate limits before download\n      await this.checkRateLimit(url);\n\n      // STEP 2: Fetch content with size and timeout protection\n      const content = await this.fetchWithLimits(url, maxSize, timeout, options.headers);\n\n      // STEP 3: Validate content type if specified\n      if (options.expectedContentType) {\n        await this.validateContentType(content, options.expectedContentType);\n      }\n\n      // STEP 4: Validate checksum if provided\n      if (options.expectedChecksum) {\n        await this.validateChecksum(content, options.expectedChecksum);\n      }\n\n      // STEP 5: Run built-in security validation\n      const securityResult = SecurityContentValidator.validateAndSanitize(content);\n      if (!securityResult.isValid && securityResult.severity === 'critical') {\n        throw DownloadError.securityError(\n          `Critical security threat detected: ${securityResult.detectedPatterns?.join(', ')}`\n        );\n      }\n\n      // STEP 6: Run custom validator if provided\n      if (options.validator) {\n        logger.debug('Running custom content validation');\n        const validationResult = await options.validator(content);\n        if (!validationResult.isValid) {\n          throw DownloadError.validationError(\n            validationResult.errorMessage || 'Content validation failed'\n          );\n        }\n      }\n\n      logger.debug(`Content validation passed (${content.length} bytes)`);\n      return securityResult.sanitizedContent || content;\n\n    } catch (error) {\n      if (error instanceof DownloadError) {\n        throw error;\n      }\n      throw DownloadError.networkError(\n        `Failed to download content from ${url}: ${error instanceof Error ? error.message : String(error)}`,\n        error instanceof Error ? error : undefined\n      );\n    }\n  }\n\n  /**\n   * Download large files using streaming with chunk-level validation\n   * \n   * @param url - URL to download from\n   * @param destinationPath - Local file path to save to\n   * @param options - Streaming download options\n   */\n  async downloadStream(\n    url: string,\n    destinationPath: string,\n    options: StreamDownloadOptions = {}\n  ): Promise<void> {\n    const startTime = Date.now();\n    const maxSize = options.maxSize || this.defaultMaxSize;\n    const timeout = options.timeout || this.defaultTimeout;\n\n    logger.debug(`Starting streaming download from ${url} to ${destinationPath}`);\n\n    try {\n      // SECURITY: Check rate limits before download\n      await this.checkRateLimit(url);\n\n      // SECURITY: Validate URL and destination path\n      this.validateUrl(url);\n      const validatedPath = await this.validateDestinationPath(destinationPath);\n\n      // Generate temporary file path for atomic operation\n      const tempPath = await this.getTempFilePath(validatedPath);\n\n      let downloadedSize = 0;\n      let timeoutHandle: NodeJS.Timeout | undefined;\n\n      // Create abort controller for timeout handling\n      const abortController = new AbortController();\n      timeoutHandle = setTimeout(() => {\n        abortController.abort();\n      }, timeout);\n\n      try {\n        // SECURITY: Fetch with abort signal for timeout\n        const response = await fetch(url, {\n          signal: abortController.signal,\n          headers: options.headers\n        });\n\n        if (!response.ok) {\n          throw new Error(`HTTP ${response.status}: ${response.statusText}`);\n        }\n\n        if (!response.body) {\n          throw new Error('Response body is null');\n        }\n\n        // Ensure temp directory exists\n        await fs.mkdir(path.dirname(tempPath), { recursive: true });\n\n        // Create write stream to temporary file\n        const writeStream = createWriteStream(tempPath);\n\n        // Create a transform stream for validation and size checking\n        const validationStream = new Readable({\n          async read() {\n            // This stream will be fed by the pipeline\n          }\n        });\n\n        // Set up chunk validation and size checking\n        const reader = response.body.getReader();\n        const pump = async () => {\n          try {\n            while (true) {\n              const { done, value } = await reader.read();\n              if (done) break;\n\n              // SECURITY: Check size limit\n              downloadedSize += value.length;\n              if (downloadedSize > maxSize) {\n                throw DownloadError.securityError(\n                  `File size exceeds limit: ${downloadedSize} > ${maxSize} bytes`\n                );\n              }\n\n              // SECURITY: Run chunk validator if provided\n              if (options.streamValidator && !options.streamValidator(value)) {\n                throw DownloadError.validationError('Chunk validation failed');\n              }\n\n              validationStream.push(value);\n            }\n            validationStream.push(null); // End stream\n          } catch (error) {\n            validationStream.destroy(error instanceof Error ? error : new Error(String(error)));\n          }\n        };\n\n        // Start the pump and pipeline concurrently\n        const [pumpResult] = await Promise.all([\n          pump(),\n          pipeline(validationStream, writeStream)\n        ]);\n\n        // Clear timeout\n        if (timeoutHandle) {\n          clearTimeout(timeoutHandle);\n          timeoutHandle = undefined;\n        }\n\n        // SECURITY: Atomic rename to final destination\n        await fs.rename(tempPath, validatedPath);\n\n        const duration = Date.now() - startTime;\n        logger.info(`Streaming download completed: ${destinationPath} (${downloadedSize} bytes, ${duration}ms)`);\n\n        // Log successful streaming download\n        SecurityMonitor.logSecurityEvent({\n          type: 'FILE_COPIED',\n          severity: 'LOW',\n          source: 'secure_downloader',\n          details: `Streamed ${downloadedSize} bytes from ${url} to ${destinationPath}`,\n          metadata: {\n            url,\n            destinationPath,\n            contentLength: downloadedSize,\n            duration\n          }\n        });\n\n      } catch (error) {\n        // SECURITY: Guaranteed cleanup of temporary file\n        try {\n          await fs.unlink(tempPath);\n          logger.debug(`Cleaned up temp file: ${tempPath}`);\n        } catch (cleanupError) {\n          logger.warn(`Failed to clean up temp file ${tempPath}: ${cleanupError}`);\n        }\n        throw error;\n      } finally {\n        if (timeoutHandle) {\n          clearTimeout(timeoutHandle);\n        }\n      }\n\n    } catch (error) {\n      const duration = Date.now() - startTime;\n      logger.error(`Streaming download failed: ${error instanceof Error ? error.message : String(error)}`);\n\n      // Log failed streaming download\n      SecurityMonitor.logSecurityEvent({\n        type: 'PATH_TRAVERSAL_ATTEMPT',\n        severity: 'MEDIUM',\n        source: 'secure_downloader',\n        details: `Streaming download failed: ${error instanceof Error ? error.message : String(error)}`,\n        metadata: {\n          url,\n          destinationPath,\n          duration,\n          errorType: error instanceof DownloadError ? error.code : 'UNKNOWN'\n        }\n      });\n\n      if (error instanceof Error && error.name === 'AbortError') {\n        throw DownloadError.timeoutError(`Download timed out after ${timeout}ms`);\n      }\n\n      if (error instanceof DownloadError) {\n        throw error;\n      }\n\n      throw DownloadError.networkError(\n        `Streaming download failed: ${error instanceof Error ? error.message : String(error)}`,\n        error instanceof Error ? error : undefined\n      );\n    }\n  }\n\n  /**\n   * Validate URL format and security with Unicode normalization\n   */\n  private validateUrl(url: string): void {\n    if (!url || typeof url !== 'string') {\n      throw DownloadError.validationError('URL must be a non-empty string');\n    }\n\n    // SECURITY FIX: DMCP-SEC-004 - Unicode normalization on user input\n    const unicodeValidation = UnicodeValidator.normalize(url);\n    const normalizedUrl = unicodeValidation.normalizedContent;\n    \n    if (!unicodeValidation.isValid) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'UNICODE_VALIDATION_ERROR',\n        severity: 'MEDIUM',\n        source: 'secure_downloader',\n        details: `URL contains suspicious Unicode patterns: ${unicodeValidation.detectedIssues?.join(', ')}`,\n        metadata: { originalUrl: url, normalizedUrl }\n      });\n    }\n    \n    // Use normalized URL for further validation\n    url = normalizedUrl;\n\n    let parsedUrl: URL;\n    try {\n      parsedUrl = new URL(url);\n    } catch (error) {\n      throw DownloadError.validationError(`Invalid URL format: ${url}`);\n    }\n\n    // SECURITY: Only allow HTTPS and HTTP protocols\n    if (!['https:', 'http:'].includes(parsedUrl.protocol)) {\n      throw DownloadError.securityError(`Unsupported protocol: ${parsedUrl.protocol}. Only HTTP/HTTPS allowed.`);\n    }\n\n    // SECURITY: Prevent requests to localhost/private networks\n    const hostname = parsedUrl.hostname.toLowerCase();\n    if (hostname === 'localhost' || hostname === '127.0.0.1' || hostname === '::1') {\n      throw DownloadError.securityError('Downloads from localhost are not allowed');\n    }\n\n    // SECURITY: Check for private IP ranges (basic protection)\n    if (hostname.startsWith('192.168.') || hostname.startsWith('10.') || hostname.startsWith('172.')) {\n      throw DownloadError.securityError('Downloads from private IP ranges are not allowed');\n    }\n  }\n\n  /**\n   * Validate destination path for security\n   */\n  private async validateDestinationPath(filePath: string): Promise<string> {\n    try {\n      // Use existing PathValidator for comprehensive path validation\n      return await PathValidator.validatePersonaPath(filePath);\n    } catch (error) {\n      throw DownloadError.securityError(\n        `Invalid destination path: ${error instanceof Error ? error.message : String(error)}`\n      );\n    }\n  }\n\n  /**\n   * Fetch content with size and timeout limits\n   */\n  private async fetchWithLimits(\n    url: string,\n    maxSize: number,\n    timeout: number,\n    headers?: Record<string, string>\n  ): Promise<string> {\n    const abortController = new AbortController();\n    const timeoutHandle = setTimeout(() => abortController.abort(), timeout);\n\n    try {\n      const response = await fetch(url, {\n        signal: abortController.signal,\n        headers: {\n          'User-Agent': 'DollhouseMCP-SecureDownloader/1.0',\n          ...headers\n        }\n      });\n\n      if (!response.ok) {\n        throw new Error(`HTTP ${response.status}: ${response.statusText}`);\n      }\n\n      // SECURITY: Check Content-Length header if available\n      const contentLength = response.headers.get('content-length');\n      if (contentLength && parseInt(contentLength, 10) > maxSize) {\n        throw DownloadError.securityError(\n          `Content size ${contentLength} exceeds limit of ${maxSize} bytes`\n        );\n      }\n\n      // Read content with size checking\n      const chunks: Uint8Array[] = [];\n      let totalSize = 0;\n\n      if (!response.body) {\n        throw new Error('Response body is null');\n      }\n\n      const reader = response.body.getReader();\n      try {\n        while (true) {\n          const { done, value } = await reader.read();\n          if (done) break;\n\n          totalSize += value.length;\n          if (totalSize > maxSize) {\n            throw DownloadError.securityError(\n              `Content size ${totalSize} exceeds limit of ${maxSize} bytes`\n            );\n          }\n\n          chunks.push(value);\n        }\n      } finally {\n        reader.releaseLock();\n      }\n\n      // Combine chunks and decode\n      const totalLength = chunks.reduce((sum, chunk) => sum + chunk.length, 0);\n      const combined = new Uint8Array(totalLength);\n      let offset = 0;\n      for (const chunk of chunks) {\n        combined.set(chunk, offset);\n        offset += chunk.length;\n      }\n\n      return new TextDecoder('utf-8').decode(combined);\n\n    } catch (error) {\n      if (error instanceof Error && error.name === 'AbortError') {\n        throw DownloadError.timeoutError(`Request timed out after ${timeout}ms`);\n      }\n      throw error;\n    } finally {\n      clearTimeout(timeoutHandle);\n    }\n  }\n\n  /**\n   * Validate content type if specified\n   */\n  private async validateContentType(content: string, expectedType: string): Promise<void> {\n    // Basic content type validation based on content analysis\n    switch (expectedType.toLowerCase()) {\n      case 'json':\n        try {\n          JSON.parse(content);\n        } catch {\n          throw DownloadError.validationError('Content is not valid JSON');\n        }\n        break;\n      case 'yaml':\n      case 'yml':\n        // Use existing YAML validation\n        if (!SecurityContentValidator.validateYamlContent(content)) {\n          throw DownloadError.validationError('Content is not valid YAML');\n        }\n        break;\n      case 'markdown':\n      case 'md':\n        // Basic markdown validation (check for frontmatter format)\n        if (content.startsWith('---')) {\n          const frontmatterEnd = content.indexOf('\\n---\\n', 3);\n          if (frontmatterEnd === -1) {\n            throw DownloadError.validationError('Invalid markdown frontmatter format');\n          }\n        }\n        break;\n      default:\n        logger.debug(`No specific validation for content type: ${expectedType}`);\n    }\n  }\n\n  /**\n   * Atomic file write using FileLockManager\n   */\n  private async atomicWriteFile(filePath: string, content: string): Promise<void> {\n    const resource = `download:${filePath}`;\n    \n    await FileLockManager.withLock(resource, async () => {\n      // Ensure directory exists\n      await fs.mkdir(path.dirname(filePath), { recursive: true });\n      \n      // Use FileLockManager's atomic write\n      await FileLockManager.atomicWriteFile(filePath, content, { encoding: 'utf-8' });\n    });\n  }\n\n  /**\n   * Direct file write (non-atomic, for when atomic is disabled)\n   */\n  private async directWriteFile(filePath: string, content: string): Promise<void> {\n    // Ensure directory exists\n    await fs.mkdir(path.dirname(filePath), { recursive: true });\n    \n    // Direct write\n    await fs.writeFile(filePath, content, 'utf-8');\n  }\n\n  /**\n   * Generate temporary file path for atomic operations\n   */\n  private async getTempFilePath(originalPath: string): Promise<string> {\n    const dir = path.dirname(originalPath);\n    const basename = path.basename(originalPath);\n    const random = randomBytes(8).toString('hex');\n    const tempDir = path.join(dir, this.tempDir);\n    \n    // Ensure temp directory exists\n    await fs.mkdir(tempDir, { recursive: true });\n    \n    return path.join(tempDir, `${basename}.${random}.tmp`);\n  }\n\n  /**\n   * Check rate limits for downloads\n   */\n  private async checkRateLimit(url: string): Promise<void> {\n    // Check global rate limit\n    const globalStatus = this.globalRateLimiter.checkLimit();\n    if (!globalStatus.allowed) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'RATE_LIMIT_EXCEEDED',\n        severity: 'MEDIUM',\n        source: 'secure_downloader',\n        details: `Global download rate limit exceeded. Retry after ${globalStatus.retryAfterMs}ms`,\n        metadata: { url, retryAfterMs: globalStatus.retryAfterMs }\n      });\n      throw DownloadError.securityError(\n        `Download rate limit exceeded. Please retry after ${Math.ceil(globalStatus.retryAfterMs! / 1000)} seconds`\n      );\n    }\n\n    // Check per-URL rate limit\n    const parsedUrl = new URL(url);\n    const urlKey = `${parsedUrl.hostname}:${parsedUrl.port || (parsedUrl.protocol === 'https:' ? '443' : '80')}`;\n    \n    if (!this.urlRateLimiters.has(urlKey)) {\n      this.urlRateLimiters.set(urlKey, new RateLimiter({\n        maxRequests: 10, // 10 requests per hour per URL\n        windowMs: 60 * 60 * 1000,\n        minDelayMs: 5000 // 5 second minimum delay between requests to same URL\n      }));\n    }\n    \n    const urlLimiter = this.urlRateLimiters.get(urlKey)!;\n    const urlStatus = urlLimiter.checkLimit();\n    if (!urlStatus.allowed) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'RATE_LIMIT_EXCEEDED',\n        severity: 'MEDIUM',\n        source: 'secure_downloader',\n        details: `Per-URL download rate limit exceeded for ${urlKey}. Retry after ${urlStatus.retryAfterMs}ms`,\n        metadata: { url, urlKey, retryAfterMs: urlStatus.retryAfterMs }\n      });\n      throw DownloadError.securityError(\n        `Too many requests to ${urlKey}. Please retry after ${Math.ceil(urlStatus.retryAfterMs! / 1000)} seconds`\n      );\n    }\n\n    // Consume rate limit tokens\n    this.globalRateLimiter.consumeToken();\n    urlLimiter.consumeToken();\n  }\n\n  /**\n   * Validate content checksum for integrity verification\n   */\n  private async validateChecksum(content: string, expectedChecksum: string): Promise<void> {\n    const normalizedExpected = expectedChecksum.toLowerCase().trim();\n    \n    // Validate checksum format (SHA-256 should be 64 hex characters)\n    if (!/^[a-f0-9]{64}$/.test(normalizedExpected)) {\n      throw DownloadError.validationError('Invalid checksum format. Expected SHA-256 (64 hex characters)');\n    }\n\n    const contentBuffer = Buffer.from(content, 'utf-8');\n    const actualChecksum = createHash('sha256').update(contentBuffer).digest('hex');\n    \n    if (actualChecksum !== normalizedExpected) {\n      SecurityMonitor.logSecurityEvent({\n        type: 'CONTENT_INJECTION_ATTEMPT',\n        severity: 'HIGH',\n        source: 'secure_downloader',\n        details: `Checksum mismatch detected - possible content tampering`,\n        metadata: { \n          expectedChecksum: normalizedExpected,\n          actualChecksum,\n          contentLength: content.length\n        }\n      });\n      throw DownloadError.securityError(\n        `Content checksum verification failed. Expected: ${normalizedExpected}, Got: ${actualChecksum}`\n      );\n    }\n\n    logger.debug(`Checksum validation passed: ${actualChecksum}`);\n  }\n\n  /**\n   * Create a content validator that combines multiple validators\n   */\n  static combineValidators(...validators: ContentValidatorFunction[]): ContentValidatorFunction {\n    return async (content: string): Promise<ValidationResult> => {\n      for (const validator of validators) {\n        const result = await validator(content);\n        if (!result.isValid) {\n          return result;\n        }\n      }\n      return { isValid: true };\n    };\n  }\n\n  /**\n   * Create a content validator for JSON content\n   */\n  static jsonValidator(): ContentValidatorFunction {\n    return async (content: string): Promise<ValidationResult> => {\n      try {\n        JSON.parse(content);\n        return { isValid: true };\n      } catch (error) {\n        return {\n          isValid: false,\n          errorMessage: `Invalid JSON: ${error instanceof Error ? error.message : String(error)}`,\n          severity: 'medium'\n        };\n      }\n    };\n  }\n\n  /**\n   * Create a content validator for YAML content\n   */\n  static yamlValidator(): ContentValidatorFunction {\n    return async (content: string): Promise<ValidationResult> => {\n      const isValid = SecurityContentValidator.validateYamlContent(content);\n      return {\n        isValid,\n        errorMessage: isValid ? undefined : 'Invalid or malicious YAML content',\n        severity: isValid ? 'low' : 'high'\n      };\n    };\n  }\n\n  /**\n   * Create a content validator for markdown content\n   */\n  static markdownValidator(): ContentValidatorFunction {\n    return async (content: string): Promise<ValidationResult> => {\n      try {\n        // Use existing persona content sanitization for markdown\n        SecurityContentValidator.sanitizePersonaContent(content);\n        return { isValid: true };\n      } catch (error) {\n        return {\n          isValid: false,\n          errorMessage: `Invalid markdown: ${error instanceof Error ? error.message : String(error)}`,\n          severity: error instanceof SecurityError ? 'critical' : 'medium'\n        };\n      }\n    };\n  }\n\n  /**\n   * Create a content validator with size limits\n   */\n  static sizeValidator(maxSize: number): ContentValidatorFunction {\n    return async (content: string): Promise<ValidationResult> => {\n      const size = Buffer.byteLength(content, 'utf-8');\n      if (size > maxSize) {\n        return {\n          isValid: false,\n          errorMessage: `Content size ${size} exceeds limit of ${maxSize} bytes`,\n          severity: 'high'\n        };\n      }\n      return { isValid: true };\n    };\n  }\n\n  /**\n   * Create a content validator that checks for forbidden patterns\n   */\n  static patternValidator(\n    forbiddenPatterns: RegExp[],\n    errorMessage: string = 'Forbidden pattern detected'\n  ): ContentValidatorFunction {\n    return async (content: string): Promise<ValidationResult> => {\n      for (const pattern of forbiddenPatterns) {\n        if (pattern.test(content)) {\n          return {\n            isValid: false,\n            errorMessage,\n            severity: 'high',\n            metadata: { pattern: pattern.source }\n          };\n        }\n      }\n      return { isValid: true };\n    };\n  }\n}"]}