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