@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
|
@@ -6,6 +6,7 @@
|
|
|
6
6
|
* It ensures users have example content to work with immediately after installation.
|
|
7
7
|
*/
|
|
8
8
|
import * as fs from 'fs/promises';
|
|
9
|
+
import * as fsSync from 'fs';
|
|
9
10
|
import * as path from 'path';
|
|
10
11
|
import { fileURLToPath } from 'url';
|
|
11
12
|
import { createHash } from 'crypto';
|
|
@@ -13,6 +14,7 @@ import { logger } from '../utils/logger.js';
|
|
|
13
14
|
import { ElementType } from './types.js';
|
|
14
15
|
import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
|
|
15
16
|
import { SecurityMonitor } from '../security/securityMonitor.js';
|
|
17
|
+
import { SecureYamlParser } from '../security/secureYamlParser.js';
|
|
16
18
|
// File operation constants
|
|
17
19
|
export const FILE_CONSTANTS = {
|
|
18
20
|
ELEMENT_EXTENSION: '.md',
|
|
@@ -24,6 +26,18 @@ export const FILE_CONSTANTS = {
|
|
|
24
26
|
FILE_PERMISSIONS: 0o644,
|
|
25
27
|
CHUNK_SIZE: 64 * 1024 // 64KB chunks for reading large files
|
|
26
28
|
};
|
|
29
|
+
// Development mode detection
|
|
30
|
+
// When running from a git clone, we don't want to auto-load test data
|
|
31
|
+
const IS_DEVELOPMENT_MODE = (() => {
|
|
32
|
+
try {
|
|
33
|
+
// Check if we're in a git repository (development mode)
|
|
34
|
+
const gitDir = path.join(process.cwd(), '.git');
|
|
35
|
+
return fsSync.existsSync(gitDir);
|
|
36
|
+
}
|
|
37
|
+
catch {
|
|
38
|
+
return false;
|
|
39
|
+
}
|
|
40
|
+
})();
|
|
27
41
|
// Internal constants
|
|
28
42
|
const DATA_DIR_CACHE_KEY = 'dollhouse_data_dir';
|
|
29
43
|
const COPY_RETRY_ATTEMPTS = 3;
|
|
@@ -33,13 +47,66 @@ export class DefaultElementProvider {
|
|
|
33
47
|
static cachedDataDir = null;
|
|
34
48
|
static populateInProgress = new Map();
|
|
35
49
|
config;
|
|
50
|
+
// PERFORMANCE OPTIMIZATION: Cache metadata with file mtime for invalidation
|
|
51
|
+
static metadataCache = new Map();
|
|
52
|
+
static MAX_CACHE_SIZE = 20; // MEMORY LEAK FIX: Further reduced cache size to prevent accumulation during performance tests
|
|
36
53
|
constructor(config) {
|
|
37
54
|
const __filename = fileURLToPath(import.meta.url);
|
|
38
55
|
this.__dirname = path.dirname(__filename);
|
|
56
|
+
// Check environment variable for test data loading
|
|
57
|
+
const envLoadTestData = process.env.DOLLHOUSE_LOAD_TEST_DATA;
|
|
58
|
+
const loadTestDataFromEnv = envLoadTestData === 'true' || envLoadTestData === '1';
|
|
59
|
+
const disableTestDataFromEnv = envLoadTestData === 'false' || envLoadTestData === '0';
|
|
60
|
+
// Check if we're in development mode (with respect to FORCE_PRODUCTION_MODE override)
|
|
61
|
+
const isDevMode = (() => {
|
|
62
|
+
// Respect FORCE_PRODUCTION_MODE override first
|
|
63
|
+
if (process.env.FORCE_PRODUCTION_MODE === 'true') {
|
|
64
|
+
return false; // Force production mode
|
|
65
|
+
}
|
|
66
|
+
if (process.env.FORCE_PRODUCTION_MODE === 'false') {
|
|
67
|
+
return true; // Force development mode
|
|
68
|
+
}
|
|
69
|
+
// Fall back to git detection
|
|
70
|
+
return IS_DEVELOPMENT_MODE;
|
|
71
|
+
})();
|
|
72
|
+
// Determine loadTestData value
|
|
73
|
+
let computedLoadTestData;
|
|
74
|
+
if (loadTestDataFromEnv) {
|
|
75
|
+
// Environment explicitly enables test data
|
|
76
|
+
computedLoadTestData = true;
|
|
77
|
+
}
|
|
78
|
+
else if (disableTestDataFromEnv) {
|
|
79
|
+
// Environment explicitly disables test data
|
|
80
|
+
computedLoadTestData = false;
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
// Default logic: enable in production, disable in development unless config overrides
|
|
84
|
+
computedLoadTestData = !isDevMode && (config?.loadTestData ?? true);
|
|
85
|
+
}
|
|
39
86
|
this.config = {
|
|
40
87
|
useDefaultPaths: true,
|
|
41
|
-
...config
|
|
88
|
+
...config,
|
|
89
|
+
// Apply final loadTestData logic - environment variables and development mode take precedence
|
|
90
|
+
loadTestData: loadTestDataFromEnv ? true : disableTestDataFromEnv ? false : computedLoadTestData
|
|
42
91
|
};
|
|
92
|
+
if (isDevMode && !this.config.loadTestData) {
|
|
93
|
+
logger.info('[DefaultElementProvider] Development mode detected - test data loading disabled');
|
|
94
|
+
logger.info('[DefaultElementProvider] To enable test data, set DOLLHOUSE_LOAD_TEST_DATA=true');
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Get the current loadTestData configuration value
|
|
99
|
+
* @returns Whether test data loading is enabled
|
|
100
|
+
*/
|
|
101
|
+
get isTestDataLoadingEnabled() {
|
|
102
|
+
return this.config.loadTestData ?? false;
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* Get whether the system is in development mode
|
|
106
|
+
* @returns Whether running in development mode
|
|
107
|
+
*/
|
|
108
|
+
get isDevelopmentMode() {
|
|
109
|
+
return IS_DEVELOPMENT_MODE;
|
|
43
110
|
}
|
|
44
111
|
/**
|
|
45
112
|
* Search paths for bundled data directory
|
|
@@ -53,9 +120,15 @@ export class DefaultElementProvider {
|
|
|
53
120
|
}
|
|
54
121
|
// Add default paths if enabled
|
|
55
122
|
if (this.config.useDefaultPaths !== false) {
|
|
123
|
+
// Skip development/repository data paths unless test data loading is enabled
|
|
124
|
+
if (this.config.loadTestData) {
|
|
125
|
+
// Development/Git installation (relative to this file)
|
|
126
|
+
paths.push(path.join(this.__dirname, '../../data'), path.join(this.__dirname, '../../../data'),
|
|
127
|
+
// Current working directory (last resort)
|
|
128
|
+
path.join(process.cwd(), 'data'));
|
|
129
|
+
}
|
|
130
|
+
// Always include NPM installation paths (these would have production data)
|
|
56
131
|
paths.push(
|
|
57
|
-
// Development/Git installation (relative to this file)
|
|
58
|
-
path.join(this.__dirname, '../../data'), path.join(this.__dirname, '../../../data'),
|
|
59
132
|
// NPM installations - macOS Homebrew
|
|
60
133
|
'/opt/homebrew/lib/node_modules/@dollhousemcp/mcp-server/data',
|
|
61
134
|
// NPM installations - standard Unix/Linux
|
|
@@ -63,9 +136,7 @@ export class DefaultElementProvider {
|
|
|
63
136
|
// NPM installations - Windows
|
|
64
137
|
'C:\\Program Files\\nodejs\\node_modules\\@dollhousemcp\\mcp-server\\data', 'C:\\Program Files (x86)\\nodejs\\node_modules\\@dollhousemcp\\mcp-server\\data',
|
|
65
138
|
// NPM installations - Windows with nvm
|
|
66
|
-
path.join(process.env.APPDATA || '', 'npm', 'node_modules', '@dollhousemcp', 'mcp-server', 'data')
|
|
67
|
-
// Current working directory (last resort)
|
|
68
|
-
path.join(process.cwd(), 'data'));
|
|
139
|
+
path.join(process.env.APPDATA || '', 'npm', 'node_modules', '@dollhousemcp', 'mcp-server', 'data'));
|
|
69
140
|
}
|
|
70
141
|
return paths;
|
|
71
142
|
}
|
|
@@ -122,6 +193,380 @@ export class DefaultElementProvider {
|
|
|
122
193
|
return false;
|
|
123
194
|
}
|
|
124
195
|
}
|
|
196
|
+
/**
|
|
197
|
+
* Validate file path to prevent path traversal attacks
|
|
198
|
+
* SECURITY FIX: Added file path validation to prevent directory traversal
|
|
199
|
+
* Previously: File paths were used without validation, allowing potential ../../../ attacks
|
|
200
|
+
* Now: Strict validation ensures paths stay within allowed directories
|
|
201
|
+
* @param filePath The file path to validate
|
|
202
|
+
* @param allowedBasePaths Array of allowed base paths (optional)
|
|
203
|
+
* @returns true if path is safe, false otherwise
|
|
204
|
+
*/
|
|
205
|
+
validateFilePath(filePath, allowedBasePaths) {
|
|
206
|
+
try {
|
|
207
|
+
// SECURITY: Normalize path to prevent traversal attempts
|
|
208
|
+
const normalizedPath = path.normalize(filePath);
|
|
209
|
+
// SECURITY: Reject paths containing traversal patterns
|
|
210
|
+
if (normalizedPath.includes('..')) {
|
|
211
|
+
logger.warn(`[DefaultElementProvider] Path traversal attempt blocked: ${filePath}`);
|
|
212
|
+
return false;
|
|
213
|
+
}
|
|
214
|
+
// SECURITY: Check for home directory expansion attempts, but allow Windows 8.3 short path names
|
|
215
|
+
// Windows short path names use ~ followed by a digit (e.g., RUNNER~1), which should be allowed
|
|
216
|
+
// Only block ~ followed by / or \ (home directory expansion patterns)
|
|
217
|
+
if (normalizedPath.includes('~/') || normalizedPath.includes('~\\')) {
|
|
218
|
+
logger.warn(`[DefaultElementProvider] Home directory expansion attempt blocked: ${filePath}`);
|
|
219
|
+
return false;
|
|
220
|
+
}
|
|
221
|
+
// SECURITY: Reject absolute paths outside allowed directories
|
|
222
|
+
if (path.isAbsolute(normalizedPath) && allowedBasePaths) {
|
|
223
|
+
const isAllowed = allowedBasePaths.some(basePath => {
|
|
224
|
+
const normalizedBase = path.normalize(basePath);
|
|
225
|
+
return normalizedPath.startsWith(normalizedBase);
|
|
226
|
+
});
|
|
227
|
+
if (!isAllowed) {
|
|
228
|
+
logger.warn(`[DefaultElementProvider] Absolute path outside allowed directories: ${filePath}`);
|
|
229
|
+
return false;
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
// SECURITY: Reject null bytes and other dangerous characters
|
|
233
|
+
if (normalizedPath.includes('\0') || normalizedPath.includes('\x00')) {
|
|
234
|
+
logger.warn(`[DefaultElementProvider] Null byte in path blocked: ${filePath}`);
|
|
235
|
+
return false;
|
|
236
|
+
}
|
|
237
|
+
return true;
|
|
238
|
+
}
|
|
239
|
+
catch (error) {
|
|
240
|
+
logger.warn(`[DefaultElementProvider] Path validation error: ${error}`);
|
|
241
|
+
return false;
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
// DEPRECATED: Commented out filename pattern detection - replaced with metadata-based detection
|
|
245
|
+
/**
|
|
246
|
+
* Cached compiled regex patterns for performance optimization
|
|
247
|
+
* @deprecated Use metadata-based detection instead
|
|
248
|
+
*/
|
|
249
|
+
// private static compiledTestPatterns: RegExp[] | null = null;
|
|
250
|
+
/**
|
|
251
|
+
* Get compiled test patterns with caching for better performance
|
|
252
|
+
* @deprecated Use metadata-based detection instead
|
|
253
|
+
* @returns Array of compiled regex patterns
|
|
254
|
+
*/
|
|
255
|
+
// private getCompiledTestPatterns(): RegExp[] {
|
|
256
|
+
// // Use cached patterns if available
|
|
257
|
+
// if (DefaultElementProvider.compiledTestPatterns) {
|
|
258
|
+
// return DefaultElementProvider.compiledTestPatterns;
|
|
259
|
+
// }
|
|
260
|
+
//
|
|
261
|
+
// // Compile and cache patterns on first use
|
|
262
|
+
// // CRITICAL FIX: Removed overly broad /^test-/i pattern that was blocking legitimate use
|
|
263
|
+
// // Users should be able to create personas like "test-driven-developer" or "test-automation-expert"
|
|
264
|
+
// // We only block specific test patterns that are clearly from our test suite
|
|
265
|
+
// DefaultElementProvider.compiledTestPatterns = [
|
|
266
|
+
// /^testpersona/i, // Our test suite pattern
|
|
267
|
+
// /^yamltest/i, // Security test pattern
|
|
268
|
+
// /^yamlbomb/i, // Security test pattern
|
|
269
|
+
// /^memory-test-/i, // Performance test pattern
|
|
270
|
+
// /^perf-test-/i, // Performance test pattern
|
|
271
|
+
// /^test-fixture-/i, // Test fixture pattern (more specific)
|
|
272
|
+
// /^test-data-/i, // Test data pattern (more specific)
|
|
273
|
+
// /bin-sh|rm-rf|pwned/i, // Malicious patterns
|
|
274
|
+
// /concurrent-\d+/i, // Concurrent test pattern
|
|
275
|
+
// /legacy\.md$/i, // Legacy test pattern
|
|
276
|
+
// /performance-test/i, // Performance test pattern
|
|
277
|
+
// /-\d{13}-[a-z0-9]+\.md$/i, // Timestamp-based test files
|
|
278
|
+
// /^unittest-/i, // Unit test pattern
|
|
279
|
+
// /^integrationtest-/i, // Integration test pattern
|
|
280
|
+
// ];
|
|
281
|
+
//
|
|
282
|
+
// return DefaultElementProvider.compiledTestPatterns;
|
|
283
|
+
// }
|
|
284
|
+
/**
|
|
285
|
+
* Check if a filename matches test data patterns that should never be copied to production
|
|
286
|
+
* @deprecated Use isDollhouseMCPTestElement() for metadata-based detection instead
|
|
287
|
+
* @param filename The filename to check
|
|
288
|
+
* @returns true if the filename matches test patterns that should be blocked
|
|
289
|
+
*/
|
|
290
|
+
// private isTestDataPattern(filename: string): boolean {
|
|
291
|
+
// const patterns = this.getCompiledTestPatterns();
|
|
292
|
+
// return patterns.some(pattern => pattern.test(filename));
|
|
293
|
+
// }
|
|
294
|
+
/**
|
|
295
|
+
* Read metadata from YAML frontmatter only (never reads content body)
|
|
296
|
+
* Uses a small buffer to safely extract only the frontmatter between --- markers
|
|
297
|
+
* @param filePath Path to the file to read metadata from
|
|
298
|
+
* @returns Parsed metadata object or null if no frontmatter found
|
|
299
|
+
*/
|
|
300
|
+
// PERFORMANCE OPTIMIZATION: Reusable buffer pool to reduce allocations
|
|
301
|
+
static bufferPool = [];
|
|
302
|
+
static MAX_POOL_SIZE = 20; // MEMORY LEAK FIX: Reduced buffer pool size to match cache size for consistent memory management
|
|
303
|
+
static bufferPoolStats = { hits: 0, misses: 0, created: 0 };
|
|
304
|
+
getBuffer() {
|
|
305
|
+
// PERFORMANCE: Track buffer pool usage for optimization monitoring
|
|
306
|
+
let buffer = DefaultElementProvider.bufferPool.pop();
|
|
307
|
+
if (buffer) {
|
|
308
|
+
DefaultElementProvider.bufferPoolStats.hits++;
|
|
309
|
+
return buffer;
|
|
310
|
+
}
|
|
311
|
+
else {
|
|
312
|
+
DefaultElementProvider.bufferPoolStats.misses++;
|
|
313
|
+
DefaultElementProvider.bufferPoolStats.created++;
|
|
314
|
+
buffer = Buffer.alloc(4096);
|
|
315
|
+
logger.debug(`[DefaultElementProvider] Created new buffer (pool empty), total created: ${DefaultElementProvider.bufferPoolStats.created}`);
|
|
316
|
+
return buffer;
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
releaseBuffer(buffer) {
|
|
320
|
+
// CRITICAL FIX: Always attempt to return buffer to pool for reuse
|
|
321
|
+
if (DefaultElementProvider.bufferPool.length < DefaultElementProvider.MAX_POOL_SIZE) {
|
|
322
|
+
buffer.fill(0); // SECURITY: Clear buffer before reuse to prevent data leakage
|
|
323
|
+
DefaultElementProvider.bufferPool.push(buffer);
|
|
324
|
+
}
|
|
325
|
+
else {
|
|
326
|
+
// PERFORMANCE: If pool is full, clear the buffer to help GC
|
|
327
|
+
buffer.fill(0);
|
|
328
|
+
logger.debug('[DefaultElementProvider] Buffer pool full, discarding buffer');
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
/**
|
|
332
|
+
* Clean up buffer pool and cache to free memory
|
|
333
|
+
* PERFORMANCE FIX: Added cleanup method to prevent memory leaks
|
|
334
|
+
* This should be called during application shutdown or periodic cleanup
|
|
335
|
+
*/
|
|
336
|
+
static cleanup() {
|
|
337
|
+
// Clear buffer pool
|
|
338
|
+
DefaultElementProvider.bufferPool.length = 0;
|
|
339
|
+
// Clear metadata cache
|
|
340
|
+
DefaultElementProvider.metadataCache.clear();
|
|
341
|
+
// Clear cached data directory
|
|
342
|
+
DefaultElementProvider.cachedDataDir = null;
|
|
343
|
+
// Clear population promises
|
|
344
|
+
DefaultElementProvider.populateInProgress.clear();
|
|
345
|
+
logger.info('[DefaultElementProvider] Memory cleanup completed', {
|
|
346
|
+
bufferStats: DefaultElementProvider.bufferPoolStats,
|
|
347
|
+
cacheCleared: true
|
|
348
|
+
});
|
|
349
|
+
// Reset stats
|
|
350
|
+
DefaultElementProvider.bufferPoolStats = { hits: 0, misses: 0, created: 0 };
|
|
351
|
+
}
|
|
352
|
+
/**
|
|
353
|
+
* Get performance statistics for monitoring
|
|
354
|
+
* PERFORMANCE MONITORING: Added statistics method for performance tracking
|
|
355
|
+
* This provides insights into buffer pool efficiency and cache performance
|
|
356
|
+
* @returns Object containing performance metrics
|
|
357
|
+
*/
|
|
358
|
+
static getPerformanceStats() {
|
|
359
|
+
const bufferHits = DefaultElementProvider.bufferPoolStats.hits;
|
|
360
|
+
const bufferMisses = DefaultElementProvider.bufferPoolStats.misses;
|
|
361
|
+
const totalRequests = bufferHits + bufferMisses;
|
|
362
|
+
return {
|
|
363
|
+
bufferPool: {
|
|
364
|
+
hits: bufferHits,
|
|
365
|
+
misses: bufferMisses,
|
|
366
|
+
created: DefaultElementProvider.bufferPoolStats.created,
|
|
367
|
+
hitRate: totalRequests > 0 ? bufferHits / totalRequests : 0,
|
|
368
|
+
poolSize: DefaultElementProvider.bufferPool.length,
|
|
369
|
+
maxPoolSize: DefaultElementProvider.MAX_POOL_SIZE
|
|
370
|
+
},
|
|
371
|
+
metadataCache: {
|
|
372
|
+
size: DefaultElementProvider.metadataCache.size,
|
|
373
|
+
maxSize: DefaultElementProvider.MAX_CACHE_SIZE
|
|
374
|
+
}
|
|
375
|
+
};
|
|
376
|
+
}
|
|
377
|
+
async readMetadataOnly(filePath, retries = 2) {
|
|
378
|
+
// PERFORMANCE: Check cache first before reading file
|
|
379
|
+
try {
|
|
380
|
+
const stats = await fs.stat(filePath);
|
|
381
|
+
const cacheKey = filePath;
|
|
382
|
+
const cached = DefaultElementProvider.metadataCache.get(cacheKey);
|
|
383
|
+
// Return cached metadata if file hasn't changed
|
|
384
|
+
// CRITICAL FIX: Use integer mtime comparison to avoid floating-point precision issues
|
|
385
|
+
if (cached && Math.floor(cached.mtime) === Math.floor(stats.mtimeMs) && cached.size === stats.size) {
|
|
386
|
+
logger.debug(`[DefaultElementProvider] Cache hit for ${filePath}`);
|
|
387
|
+
return cached.metadata;
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
catch {
|
|
391
|
+
// File doesn't exist, proceed with normal flow
|
|
392
|
+
}
|
|
393
|
+
try {
|
|
394
|
+
// Open file and read only first 4KB to avoid reading dangerous content
|
|
395
|
+
const fd = await fs.open(filePath, 'r');
|
|
396
|
+
// PERFORMANCE: Use buffer pool instead of allocating new buffer each time
|
|
397
|
+
const buffer = this.getBuffer();
|
|
398
|
+
try {
|
|
399
|
+
const result = await fd.read(buffer, 0, 4096, 0);
|
|
400
|
+
const header = buffer.subarray(0, result.bytesRead).toString('utf-8');
|
|
401
|
+
// Look for YAML frontmatter between --- markers
|
|
402
|
+
// Support both Unix (\n) and Windows (\r\n) line endings
|
|
403
|
+
const match = header.match(/^---\r?\n([\s\S]*?)\r?\n---/);
|
|
404
|
+
if (!match) {
|
|
405
|
+
return null; // No frontmatter found
|
|
406
|
+
}
|
|
407
|
+
// Parse the YAML frontmatter safely
|
|
408
|
+
try {
|
|
409
|
+
// SECURITY FIX: Replace direct YAML parsing function with SecureYamlParser for enhanced security
|
|
410
|
+
// SecureYamlParser provides additional validation, injection prevention, and content sanitization
|
|
411
|
+
// It expects full YAML with --- markers, so we reconstruct the frontmatter block
|
|
412
|
+
// We disable specific field validation as this is general metadata parsing, not persona-specific
|
|
413
|
+
const fullYaml = `---\n${match[1]}\n---`;
|
|
414
|
+
const parseResult = SecureYamlParser.parse(fullYaml, {
|
|
415
|
+
validateContent: false,
|
|
416
|
+
validateFields: false
|
|
417
|
+
});
|
|
418
|
+
const metadata = parseResult.data;
|
|
419
|
+
// PERFORMANCE: Cache the metadata with file stats for future reads
|
|
420
|
+
if (typeof metadata === 'object' && metadata !== null) {
|
|
421
|
+
try {
|
|
422
|
+
const stats = await fs.stat(filePath);
|
|
423
|
+
const cacheEntry = {
|
|
424
|
+
metadata,
|
|
425
|
+
mtime: stats.mtimeMs,
|
|
426
|
+
size: stats.size
|
|
427
|
+
};
|
|
428
|
+
// CRITICAL MEMORY LEAK FIX: More aggressive cache management to prevent unbounded growth
|
|
429
|
+
// Check if this entry already exists and just update it instead of adding new
|
|
430
|
+
if (DefaultElementProvider.metadataCache.has(filePath)) {
|
|
431
|
+
// Update existing entry - no eviction needed
|
|
432
|
+
DefaultElementProvider.metadataCache.set(filePath, cacheEntry);
|
|
433
|
+
logger.debug(`[DefaultElementProvider] Updated existing cache entry for ${filePath}`);
|
|
434
|
+
}
|
|
435
|
+
else {
|
|
436
|
+
// New entry - check if we need to evict first
|
|
437
|
+
// Use > instead of >= to ensure we never exceed MAX_CACHE_SIZE
|
|
438
|
+
if (DefaultElementProvider.metadataCache.size >= DefaultElementProvider.MAX_CACHE_SIZE) {
|
|
439
|
+
// More aggressive eviction: remove enough entries to stay well under limit
|
|
440
|
+
const entriesToEvict = Math.max(1, Math.floor(DefaultElementProvider.MAX_CACHE_SIZE * 0.4));
|
|
441
|
+
const keysToEvict = Array.from(DefaultElementProvider.metadataCache.keys()).slice(0, entriesToEvict);
|
|
442
|
+
for (const key of keysToEvict) {
|
|
443
|
+
DefaultElementProvider.metadataCache.delete(key);
|
|
444
|
+
}
|
|
445
|
+
logger.debug(`[DefaultElementProvider] Evicted ${keysToEvict.length} cache entries to manage memory (cache size was ${DefaultElementProvider.metadataCache.size + keysToEvict.length})`);
|
|
446
|
+
}
|
|
447
|
+
DefaultElementProvider.metadataCache.set(filePath, cacheEntry);
|
|
448
|
+
logger.debug(`[DefaultElementProvider] Added new cache entry for ${filePath} (cache size now: ${DefaultElementProvider.metadataCache.size})`);
|
|
449
|
+
}
|
|
450
|
+
}
|
|
451
|
+
catch {
|
|
452
|
+
// Ignore cache errors, return metadata anyway
|
|
453
|
+
}
|
|
454
|
+
return metadata;
|
|
455
|
+
}
|
|
456
|
+
return null;
|
|
457
|
+
}
|
|
458
|
+
catch (yamlError) {
|
|
459
|
+
// Invalid YAML, return null
|
|
460
|
+
// ENHANCEMENT: Include error type for better debugging
|
|
461
|
+
const yamlErrorType = yamlError?.constructor?.name || 'YAMLError';
|
|
462
|
+
logger.debug(`[DefaultElementProvider] Invalid YAML in ${filePath}: ${yamlErrorType} - ${yamlError}`);
|
|
463
|
+
return null;
|
|
464
|
+
}
|
|
465
|
+
}
|
|
466
|
+
finally {
|
|
467
|
+
// CRITICAL FIX: Ensure file descriptor is closed and buffer is released in ALL paths
|
|
468
|
+
try {
|
|
469
|
+
await fd.close();
|
|
470
|
+
}
|
|
471
|
+
catch (closeError) {
|
|
472
|
+
logger.debug(`[DefaultElementProvider] Error closing file descriptor for ${filePath}: ${closeError}`);
|
|
473
|
+
}
|
|
474
|
+
// PERFORMANCE: Return buffer to pool for reuse
|
|
475
|
+
this.releaseBuffer(buffer);
|
|
476
|
+
}
|
|
477
|
+
}
|
|
478
|
+
catch (error) {
|
|
479
|
+
// ENHANCEMENT: Include error type in debug logs for better debugging
|
|
480
|
+
const errorType = error?.constructor?.name || 'UnknownError';
|
|
481
|
+
const errorCode = error?.code || 'NO_CODE';
|
|
482
|
+
// RELIABILITY: Add retry logic for transient failures
|
|
483
|
+
if (retries > 0 && (errorCode === 'EBUSY' || errorCode === 'EAGAIN')) {
|
|
484
|
+
logger.debug(`[DefaultElementProvider] Retrying read for ${filePath} after ${errorType}:${errorCode}`);
|
|
485
|
+
await new Promise(resolve => setTimeout(resolve, 50)); // Brief delay before retry
|
|
486
|
+
return this.readMetadataOnly(filePath, retries - 1);
|
|
487
|
+
}
|
|
488
|
+
logger.debug(`[DefaultElementProvider] Could not read metadata from ${filePath}: ${errorType}:${errorCode} - ${error?.message || error}`);
|
|
489
|
+
return null;
|
|
490
|
+
}
|
|
491
|
+
}
|
|
492
|
+
/**
|
|
493
|
+
* Check if a file is a DollhouseMCP test element based on metadata
|
|
494
|
+
* This replaces filename pattern detection with accurate metadata-based detection
|
|
495
|
+
* @param filePath Path to the file to check
|
|
496
|
+
* @returns true if the file contains _dollhouseMCPTest: true metadata
|
|
497
|
+
*/
|
|
498
|
+
async isDollhouseMCPTestElement(filePath) {
|
|
499
|
+
try {
|
|
500
|
+
const metadata = await this.readMetadataOnly(filePath);
|
|
501
|
+
const isTest = !!(metadata && metadata._dollhouseMCPTest === true);
|
|
502
|
+
return isTest;
|
|
503
|
+
}
|
|
504
|
+
catch (error) {
|
|
505
|
+
// If we can't read the metadata, assume it's not a test file
|
|
506
|
+
logger.debug(`[DefaultElementProvider] Error checking test metadata for ${filePath}: ${error}`);
|
|
507
|
+
return false;
|
|
508
|
+
}
|
|
509
|
+
}
|
|
510
|
+
/**
|
|
511
|
+
* Detect if we're in a production environment by checking for production indicators
|
|
512
|
+
* Uses a confidence-based approach requiring multiple indicators for better accuracy
|
|
513
|
+
* @returns true if this appears to be a production environment
|
|
514
|
+
*/
|
|
515
|
+
isProductionEnvironment() {
|
|
516
|
+
// Allow tests to explicitly override production mode detection
|
|
517
|
+
if (process.env.FORCE_PRODUCTION_MODE === 'true') {
|
|
518
|
+
return true;
|
|
519
|
+
}
|
|
520
|
+
if (process.env.FORCE_PRODUCTION_MODE === 'false') {
|
|
521
|
+
return false;
|
|
522
|
+
}
|
|
523
|
+
// Weighted indicators for production detection
|
|
524
|
+
const indicators = {
|
|
525
|
+
// Strong indicators (weight: 2)
|
|
526
|
+
hasUserHomeDir: (process.env.HOME && (process.env.HOME.includes('/Users/') || process.env.HOME.includes('/home/'))) ||
|
|
527
|
+
!!process.env.USERPROFILE,
|
|
528
|
+
isProductionNode: process.env.NODE_ENV === 'production',
|
|
529
|
+
notInTestDir: (() => {
|
|
530
|
+
const cwd = process.cwd().toLowerCase();
|
|
531
|
+
// Normalize path separators for cross-platform checking (Windows uses \ but checks use /)
|
|
532
|
+
const normalizedCwd = cwd.replace(/\\/g, '/');
|
|
533
|
+
return !normalizedCwd.includes('/test') &&
|
|
534
|
+
!normalizedCwd.includes('/__tests__') &&
|
|
535
|
+
!normalizedCwd.includes('/temp') &&
|
|
536
|
+
!normalizedCwd.includes('/dist/test');
|
|
537
|
+
})(),
|
|
538
|
+
// Moderate indicators (weight: 1)
|
|
539
|
+
notInCI: !process.env.CI,
|
|
540
|
+
noTestEnv: process.env.NODE_ENV !== 'test',
|
|
541
|
+
noDevEnv: process.env.NODE_ENV !== 'development',
|
|
542
|
+
};
|
|
543
|
+
// Calculate weighted score
|
|
544
|
+
let score = 0;
|
|
545
|
+
if (indicators.hasUserHomeDir)
|
|
546
|
+
score += 2;
|
|
547
|
+
if (indicators.isProductionNode)
|
|
548
|
+
score += 2;
|
|
549
|
+
if (indicators.notInTestDir)
|
|
550
|
+
score += 2;
|
|
551
|
+
if (indicators.notInCI)
|
|
552
|
+
score += 1;
|
|
553
|
+
if (indicators.noTestEnv)
|
|
554
|
+
score += 1;
|
|
555
|
+
if (indicators.noDevEnv)
|
|
556
|
+
score += 1;
|
|
557
|
+
// Log detection details for debugging
|
|
558
|
+
const activeIndicators = Object.entries(indicators)
|
|
559
|
+
.filter(([_, value]) => value)
|
|
560
|
+
.map(([key]) => key);
|
|
561
|
+
// TYPESCRIPT FIX: Removed logger.isDebugEnabled() check as this method doesn't exist on MCPLogger
|
|
562
|
+
// The logger already handles debug level internally, so we can call debug() directly
|
|
563
|
+
if (score >= 3) {
|
|
564
|
+
logger.debug('[DefaultElementProvider] Production environment detected', { score, activeIndicators, forceMode: 'not set' });
|
|
565
|
+
}
|
|
566
|
+
// Require a score of at least 3 for production detection (more confident)
|
|
567
|
+
// This prevents false positives in edge cases while maintaining security
|
|
568
|
+
return score >= 3;
|
|
569
|
+
}
|
|
125
570
|
/**
|
|
126
571
|
* Copy all files from source directory to destination directory
|
|
127
572
|
* Skips files that already exist to preserve user modifications
|
|
@@ -146,6 +591,40 @@ export class DefaultElementProvider {
|
|
|
146
591
|
}
|
|
147
592
|
const sourcePath = path.join(sourceDir, normalizedFile.normalizedContent);
|
|
148
593
|
const destPath = path.join(destDir, normalizedFile.normalizedContent);
|
|
594
|
+
// SECURITY FIX: Validate file paths to prevent path traversal attacks
|
|
595
|
+
// This prevents malicious files from escaping the intended directory structure
|
|
596
|
+
const sourceValid = this.validateFilePath(sourcePath, [sourceDir]);
|
|
597
|
+
const destValid = this.validateFilePath(destPath, [destDir]);
|
|
598
|
+
if (!sourceValid || !destValid) {
|
|
599
|
+
logger.warn(`[DefaultElementProvider] Skipping file with invalid path: ${normalizedFile.normalizedContent}`, { sourcePath, destPath, elementType });
|
|
600
|
+
continue;
|
|
601
|
+
}
|
|
602
|
+
// Production safety check: Block DollhouseMCP test elements in production environments
|
|
603
|
+
// Skip this check if loadTestData is explicitly enabled (for testing scenarios)
|
|
604
|
+
if (!this.config.loadTestData && this.isProductionEnvironment()) {
|
|
605
|
+
const isDollhouseTest = await this.isDollhouseMCPTestElement(sourcePath);
|
|
606
|
+
if (isDollhouseTest) {
|
|
607
|
+
logger.warn(`[DefaultElementProvider] SECURITY: Blocking DollhouseMCP test element in production: ${normalizedFile.normalizedContent}`, {
|
|
608
|
+
file: normalizedFile.normalizedContent,
|
|
609
|
+
reason: 'DollhouseMCP test element detected in production environment',
|
|
610
|
+
elementType
|
|
611
|
+
});
|
|
612
|
+
// Log security event for blocked test data
|
|
613
|
+
SecurityMonitor.logSecurityEvent({
|
|
614
|
+
type: 'TEST_DATA_BLOCKED',
|
|
615
|
+
severity: 'MEDIUM',
|
|
616
|
+
source: 'DefaultElementProvider.copyElementFiles',
|
|
617
|
+
details: `Blocked DollhouseMCP test element in production: ${normalizedFile.normalizedContent}`,
|
|
618
|
+
metadata: {
|
|
619
|
+
filename: normalizedFile.normalizedContent,
|
|
620
|
+
elementType,
|
|
621
|
+
reason: 'DollhouseMCP test element detected in production environment',
|
|
622
|
+
detectionMethod: 'metadata-based'
|
|
623
|
+
}
|
|
624
|
+
});
|
|
625
|
+
continue;
|
|
626
|
+
}
|
|
627
|
+
}
|
|
149
628
|
try {
|
|
150
629
|
// Check if destination file already exists
|
|
151
630
|
await fs.access(destPath);
|
|
@@ -322,6 +801,19 @@ export class DefaultElementProvider {
|
|
|
322
801
|
* @param portfolioBaseDir Base directory of the portfolio
|
|
323
802
|
*/
|
|
324
803
|
async performPopulation(portfolioBaseDir) {
|
|
804
|
+
// Check if test data loading is disabled
|
|
805
|
+
// Note: This check is needed even though constructor sets config, because
|
|
806
|
+
// config can be overridden after construction
|
|
807
|
+
// Use production environment detection that respects FORCE_PRODUCTION_MODE
|
|
808
|
+
const isDevelopmentMode = !this.isProductionEnvironment();
|
|
809
|
+
if (isDevelopmentMode && !this.config.loadTestData) {
|
|
810
|
+
logger.info('[DefaultElementProvider] Skipping default element population in development mode', {
|
|
811
|
+
portfolioBaseDir,
|
|
812
|
+
reason: 'Test data loading disabled',
|
|
813
|
+
enableWith: 'Set DOLLHOUSE_LOAD_TEST_DATA=true to enable'
|
|
814
|
+
});
|
|
815
|
+
return;
|
|
816
|
+
}
|
|
325
817
|
logger.info('[DefaultElementProvider] Starting default element population', { portfolioBaseDir });
|
|
326
818
|
// Log security event for portfolio initialization
|
|
327
819
|
SecurityMonitor.logSecurityEvent({
|
|
@@ -383,4 +875,4 @@ export class DefaultElementProvider {
|
|
|
383
875
|
}
|
|
384
876
|
}
|
|
385
877
|
}
|
|
386
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DefaultElementProvider.js","sourceRoot":"","sources":["../../src/portfolio/DefaultElementProvider.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,KAAK,EAAE,MAAM,aAAa,CAAC;AAClC,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAC7B,OAAO,EAAE,aAAa,EAAE,MAAM,KAAK,CAAC;AACpC,OAAO,EAAE,UAAU,EAAE,MAAM,QAAQ,CAAC;AACpC,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAC5C,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AACzC,OAAO,EAAE,gBAAgB,EAAE,MAAM,4CAA4C,CAAC;AAC9E,OAAO,EAAE,eAAe,EAAE,MAAM,gCAAgC,CAAC;AAEjE,2BAA2B;AAC3B,MAAM,CAAC,MAAM,cAAc,GAAG;IAC5B,iBAAiB,EAAE,KAAK;IACxB,cAAc,EAAE,OAAO;IACvB,aAAa,EAAE,MAAM;IACrB,cAAc,EAAE,OAAO;IACvB,aAAa,EAAE,EAAE,GAAG,IAAI,GAAG,IAAI,EAAE,gCAAgC;IACjE,kBAAkB,EAAE,QAAQ;IAC5B,gBAAgB,EAAE,KAAK;IACvB,UAAU,EAAE,EAAE,GAAG,IAAI,CAAC,sCAAsC;CACpD,CAAC;AAEX,qBAAqB;AACrB,MAAM,kBAAkB,GAAG,oBAAoB,CAAC;AAChD,MAAM,mBAAmB,GAAG,CAAC,CAAC;AAC9B,MAAM,gBAAgB,GAAG,GAAG,CAAC,CAAC,KAAK;AASnC,MAAM,OAAO,sBAAsB;IAChB,SAAS,CAAS;IAC3B,MAAM,CAAC,aAAa,GAAkB,IAAI,CAAC;IAC3C,MAAM,CAAC,kBAAkB,GAA+B,IAAI,GAAG,EAAE,CAAC;IACzD,MAAM,CAA+B;IAEtD,YAAY,MAAqC;QAC/C,MAAM,UAAU,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAClD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QAC1C,IAAI,CAAC,MAAM,GAAG;YACZ,eAAe,EAAE,IAAI;YACrB,GAAG,MAAM;SACV,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,IAAY,eAAe;QACzB,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,4CAA4C;QAC5C,IAAI,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE,CAAC;YAChC,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QAC7C,CAAC;QAED,+BAA+B;QAC/B,IAAI,IAAI,CAAC,MAAM,CAAC,eAAe,KAAK,KAAK,EAAE,CAAC;YAC1C,KAAK,CAAC,IAAI;YACR,uDAAuD;YACvD,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,YAAY,CAAC,EACvC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,CAAC;YAE1C,qCAAqC;YACrC,8DAA8D;YAE9D,0CAA0C;YAC1C,2DAA2D,EAC3D,qDAAqD;YAErD,8BAA8B;YAC9B,0EAA0E,EAC1E,gFAAgF;YAEhF,uCAAuC;YACvC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,IAAI,EAAE,EAAE,KAAK,EAAE,cAAc,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,CAAC;YAElG,0CAA0C;YAC1C,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,MAAM,CAAC,CACjC,CAAC;QACJ,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;OAGG;IACK,KAAK,CAAC,iBAAiB;QAC7B,mCAAmC;QACnC,IAAI,sBAAsB,CAAC,aAAa,KAAK,IAAI,EAAE,CAAC;YAClD,OAAO,sBAAsB,CAAC,aAAa,CAAC;QAC9C,CAAC;QAED,qDAAqD;QACrD,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,EAAE,UAAU,EAAE,EAAE;YAClE,IAAI,CAAC;gBACH,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;gBACxC,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC;oBACxB,6CAA6C;oBAC7C,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,UAAU,CAAC,CAAC,CAAC;oBAClF,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC,CAAC;oBAC9E,IAAI,WAAW,IAAI,SAAS,EAAE,CAAC;wBAC7B,OAAO,UAAU,CAAC;oBACpB,CAAC;gBACH,CAAC;YACH,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,+CAA+C;gBAC/C,OAAO,IAAI,CAAC;YACd,CAAC;YACD,OAAO,IAAI,CAAC;QACd,CAAC,CAAC,CAAC;QAEH,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;QAExD,mCAAmC;QACnC,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;YAC7B,IAAI,MAAM,CAAC,MAAM,KAAK,WAAW,IAAI,MAAM,CAAC,KAAK,KAAK,IAAI,EAAE,CAAC;gBAC3D,MAAM,CAAC,IAAI,CAAC,qDAAqD,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;gBACjF,mBAAmB;gBACnB,sBAAsB,CAAC,aAAa,GAAG,MAAM,CAAC,KAAK,CAAC;gBACpD,OAAO,MAAM,CAAC,KAAK,CAAC;YACtB,CAAC;QACH,CAAC;QAED,MAAM,CAAC,IAAI,CAAC,6EAA6E,CAAC,CAAC;QAC3F,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACK,KAAK,CAAC,eAAe,CAAC,OAAe;QAC3C,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACrC,OAAO,KAAK,CAAC,WAAW,EAAE,CAAC;QAC7B,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;;OAGG;IACK,KAAK,CAAC,gBAAgB,CAAC,SAAiB,EAAE,OAAe,EAAE,WAAmB;QACpF,IAAI,WAAW,GAAG,CAAC,CAAC;QAEpB,IAAI,CAAC;YACH,sCAAsC;YACtC,MAAM,EAAE,CAAC,KAAK,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;YAE7C,wBAAwB;YACxB,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;YAE1C,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,2BAA2B;gBAC3B,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,iBAAiB,CAAC,EAAE,CAAC;oBACrD,SAAS;gBACX,CAAC;gBAED,kCAAkC;gBAClC,MAAM,cAAc,GAAG,gBAAgB,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;gBACxD,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,CAAC;oBAC5B,MAAM,CAAC,IAAI,CAAC,gEAAgE,IAAI,EAAE,CAAC,CAAC;oBACpF,SAAS;gBACX,CAAC;gBAED,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,cAAc,CAAC,iBAAiB,CAAC,CAAC;gBAC1E,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,cAAc,CAAC,iBAAiB,CAAC,CAAC;gBAEtE,IAAI,CAAC;oBACH,2CAA2C;oBAC3C,MAAM,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;oBAC1B,MAAM,CAAC,KAAK,CAAC,oDAAoD,cAAc,CAAC,iBAAiB,EAAE,CAAC,CAAC;oBACrG,SAAS;gBACX,CAAC;gBAAC,MAAM,CAAC;oBACP,wCAAwC;gBAC1C,CAAC;gBAED,IAAI,CAAC;oBACH,sCAAsC;oBACtC,MAAM,WAAW,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;oBAE9C,wBAAwB;oBACxB,IAAI,WAAW,CAAC,IAAI,GAAG,cAAc,CAAC,aAAa,EAAE,CAAC;wBACpD,MAAM,CAAC,IAAI,CACT,oDAAoD,cAAc,CAAC,iBAAiB,IAAI;4BACxF,GAAG,WAAW,CAAC,IAAI,gBAAgB,cAAc,CAAC,aAAa,SAAS,EACxE;4BACE,IAAI,EAAE,cAAc,CAAC,iBAAiB;4BACtC,IAAI,EAAE,WAAW,CAAC,IAAI;4BACtB,OAAO,EAAE,cAAc,CAAC,aAAa;4BACrC,WAAW;yBACZ,CACF,CAAC;wBACF,SAAS;oBACX,CAAC;oBAED,kCAAkC;oBAClC,MAAM,IAAI,CAAC,wBAAwB,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;oBAC1D,WAAW,EAAE,CAAC;oBACd,MAAM,CAAC,KAAK,CAAC,mCAAmC,WAAW,KAAK,cAAc,CAAC,iBAAiB,EAAE,CAAC,CAAC;oBAEpG,0CAA0C;oBAC1C,eAAe,CAAC,gBAAgB,CAAC;wBAC/B,IAAI,EAAE,aAAa;wBACnB,QAAQ,EAAE,KAAK;wBACf,MAAM,EAAE,yCAAyC;wBACjD,OAAO,EAAE,kBAAkB,WAAW,UAAU,cAAc,CAAC,iBAAiB,EAAE;wBAClF,QAAQ,EAAE;4BACR,UAAU;4BACV,QAAQ;4BACR,WAAW;4BACX,QAAQ,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI;yBACzC;qBACF,CAAC,CAAC;gBACL,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACf,MAAM,GAAG,GAAG,KAAc,CAAC;oBAC3B,MAAM,CAAC,KAAK,CACV,2CAA2C,cAAc,CAAC,iBAAiB,EAAE,EAC7E;wBACE,KAAK,EAAE,GAAG,CAAC,OAAO;wBAClB,KAAK,EAAE,GAAG,CAAC,KAAK;wBAChB,UAAU;wBACV,QAAQ;wBACR,WAAW;qBACZ,CACF,CAAC;oBACF,0DAA0D;gBAC5D,CAAC;YACH,CAAC;YAED,IAAI,WAAW,GAAG,CAAC,EAAE,CAAC;gBACpB,MAAM,CAAC,IAAI,CAAC,mCAAmC,WAAW,IAAI,WAAW,UAAU,CAAC,CAAC;YACvF,CAAC;QAEH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,KAAK,CAAC,0CAA0C,WAAW,SAAS,EAAE,KAAK,CAAC,CAAC;QACtF,CAAC;QAED,OAAO,WAAW,CAAC;IACrB,CAAC;IAED;;;OAGG;IACH;;;;OAIG;IACK,KAAK,CAAC,iBAAiB,CAAC,QAAgB;QAC9C,MAAM,IAAI,GAAG,UAAU,CAAC,cAAc,CAAC,kBAAkB,CAAC,CAAC;QAC3D,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC;QAE5C,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC;YACvD,IAAI,SAAiB,CAAC;YAEtB,GAAG,CAAC;gBACF,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,EAAE,cAAc,CAAC,UAAU,CAAC,CAAC;gBACvE,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC;gBAE7B,IAAI,SAAS,GAAG,CAAC,EAAE,CAAC;oBAClB,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC,CAAC;gBAC7C,CAAC;YACH,CAAC,QAAQ,SAAS,GAAG,CAAC,EAAE;YAExB,OAAO,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC5B,CAAC;gBAAS,CAAC;YACT,MAAM,MAAM,CAAC,KAAK,EAAE,CAAC;QACvB,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACK,KAAK,CAAC,wBAAwB,CAAC,UAAkB,EAAE,QAAgB;QACzE,IAAI,SAAS,GAAiB,IAAI,CAAC;QAEnC,KAAK,IAAI,OAAO,GAAG,CAAC,EAAE,OAAO,IAAI,mBAAmB,EAAE,OAAO,EAAE,EAAE,CAAC;YAChE,IAAI,CAAC;gBACH,gBAAgB;gBAChB,MAAM,EAAE,CAAC,QAAQ,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;gBAExC,sBAAsB;gBACtB,MAAM,CAAC,WAAW,EAAE,SAAS,CAAC,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;oBACjD,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC;oBACnB,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC;iBAClB,CAAC,CAAC;gBAEH,IAAI,WAAW,CAAC,IAAI,KAAK,SAAS,CAAC,IAAI,EAAE,CAAC;oBACxC,MAAM,IAAI,KAAK,CACb,sCAAsC,WAAW,CAAC,IAAI,UAAU;wBAChE,gBAAgB,SAAS,CAAC,IAAI,QAAQ,CACvC,CAAC;gBACJ,CAAC;gBAED,iDAAiD;gBACjD,MAAM,CAAC,cAAc,EAAE,YAAY,CAAC,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC;oBACvD,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC;oBAClC,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC;iBACjC,CAAC,CAAC;gBAEH,IAAI,cAAc,KAAK,YAAY,EAAE,CAAC;oBACpC,MAAM,IAAI,KAAK,CACb,0CAA0C,cAAc,IAAI;wBAC5D,gBAAgB,YAAY,EAAE,CAC/B,CAAC;gBACJ,CAAC;gBAED,yBAAyB;gBACzB,IAAI,CAAC;oBACH,MAAM,EAAE,CAAC,KAAK,CAAC,QAAQ,EAAE,cAAc,CAAC,gBAAgB,CAAC,CAAC;gBAC5D,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACf,MAAM,CAAC,KAAK,CACV,yDAAyD,QAAQ,KAAK,KAAK,EAAE,EAC7E,EAAE,UAAU,EAAE,QAAQ,EAAE,OAAO,EAAE,CAClC,CAAC;gBACJ,CAAC;gBAED,qCAAqC;gBACrC,MAAM,CAAC,KAAK,CACV,8DAA8D,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,EACzF,EAAE,IAAI,EAAE,WAAW,CAAC,IAAI,EAAE,QAAQ,EAAE,cAAc,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,CACrE,CAAC;gBACF,OAAO;YAET,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,SAAS,GAAG,KAAc,CAAC;gBAE3B,uBAAuB;gBACvB,IAAI,CAAC;oBACH,MAAM,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;gBAC5B,CAAC;gBAAC,MAAM,CAAC;oBACP,wBAAwB;gBAC1B,CAAC;gBAED,IAAI,OAAO,GAAG,mBAAmB,EAAE,CAAC;oBAClC,MAAM,CAAC,KAAK,CACV,yCAAyC,OAAO,sBAAsB,EACtE,EAAE,KAAK,EAAE,SAAS,CAAC,OAAO,EAAE,UAAU,EAAE,QAAQ,EAAE,CACnD,CAAC;oBACF,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,gBAAgB,GAAG,OAAO,CAAC,CAAC,CAAC;gBAChF,CAAC;YACH,CAAC;QACH,CAAC;QAED,sBAAsB;QACtB,MAAM,IAAI,KAAK,CACb,kBAAkB,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,UAAU,mBAAmB,aAAa;YACrF,GAAG,SAAS,EAAE,OAAO,IAAI,eAAe,EAAE,CAC3C,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACI,KAAK,CAAC,gBAAgB,CAAC,gBAAwB;QACpD,gEAAgE;QAChE,MAAM,kBAAkB,GAAG,sBAAsB,CAAC,kBAAkB,CAAC,GAAG,CAAC,gBAAgB,CAAC,CAAC;QAC3F,IAAI,kBAAkB,EAAE,CAAC;YACvB,MAAM,CAAC,KAAK,CACV,mFAAmF,EACnF,EAAE,gBAAgB,EAAE,CACrB,CAAC;YACF,OAAO,kBAAkB,CAAC;QAC5B,CAAC;QAED,gCAAgC;QAChC,MAAM,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,CAAC,gBAAgB,CAAC;aAC/D,OAAO,CAAC,GAAG,EAAE;YACZ,qBAAqB;YACrB,sBAAsB,CAAC,kBAAkB,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC;QACrE,CAAC,CAAC,CAAC;QAEL,sBAAsB,CAAC,kBAAkB,CAAC,GAAG,CAAC,gBAAgB,EAAE,iBAAiB,CAAC,CAAC;QACnF,OAAO,iBAAiB,CAAC;IAC3B,CAAC;IAED;;;OAGG;IACK,KAAK,CAAC,iBAAiB,CAAC,gBAAwB;QACtD,MAAM,CAAC,IAAI,CACT,8DAA8D,EAC9D,EAAE,gBAAgB,EAAE,CACrB,CAAC;QAEF,kDAAkD;QAClD,eAAe,CAAC,gBAAgB,CAAC;YAC/B,IAAI,EAAE,0BAA0B;YAChC,QAAQ,EAAE,KAAK;YACf,MAAM,EAAE,0CAA0C;YAClD,OAAO,EAAE,sDAAsD,gBAAgB,EAAE;SAClF,CAAC,CAAC;QAEH,kCAAkC;QAClC,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/C,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,MAAM,CAAC,IAAI,CACT,uFAAuF,EACvF;gBACE,WAAW,EAAE,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,oCAAoC;gBACnF,GAAG,EAAE,OAAO,CAAC,GAAG,EAAE;gBAClB,OAAO,EAAE,IAAI,CAAC,SAAS;aACxB,CACF,CAAC;YACF,OAAO;QACT,CAAC;QAED,2BAA2B;QAC3B,IAAI,WAAW,GAAG,CAAC,CAAC;QACpB,MAAM,YAAY,GAA2B,EAAE,CAAC;QAEhD,0EAA0E;QAC1E,KAAK,MAAM,WAAW,IAAI,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;YACrD,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,WAAW,CAAC,CAAC;YAClD,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,WAAW,CAAC,CAAC;YAEzD,IAAI,CAAC;gBACH,mCAAmC;gBACnC,MAAM,EAAE,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;gBAC3B,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAE,OAAO,EAAE,WAAW,CAAC,CAAC;gBACjF,YAAY,CAAC,WAAW,CAAC,GAAG,WAAW,CAAC;gBACxC,WAAW,IAAI,WAAW,CAAC;YAC7B,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,uCAAuC;gBACvC,MAAM,CAAC,KAAK,CAAC,+BAA+B,WAAW,4BAA4B,CAAC,CAAC;YACvF,CAAC;QACH,CAAC;QAED,IAAI,WAAW,GAAG,CAAC,EAAE,CAAC;YACpB,MAAM,CAAC,IAAI,CACT,kEAAkE,WAAW,qBAAqB,EAClG;gBACE,gBAAgB;gBAChB,OAAO;gBACP,SAAS,EAAE,YAAY;aACxB,CACF,CAAC;YAEF,+CAA+C;YAC/C,eAAe,CAAC,gBAAgB,CAAC;gBAC/B,IAAI,EAAE,qBAAqB;gBAC3B,QAAQ,EAAE,KAAK;gBACf,MAAM,EAAE,0CAA0C;gBAClD,OAAO,EAAE,yCAAyC,WAAW,mBAAmB;gBAChF,QAAQ,EAAE;oBACR,gBAAgB;oBAChB,OAAO;oBACP,YAAY;iBACb;aACF,CAAC,CAAC;QACL,CAAC;aAAM,CAAC;YACN,MAAM,CAAC,IAAI,CAAC,uFAAuF,CAAC,CAAC;QACvG,CAAC;IACH,CAAC","sourcesContent":["/**\n * DefaultElementProvider - Populates portfolio with default elements from bundled data\n * \n * This class handles copying default personas, skills, templates, and other elements\n * from the NPM package or Git repository to the user's portfolio on first run.\n * It ensures users have example content to work with immediately after installation.\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { fileURLToPath } from 'url';\nimport { createHash } from 'crypto';\nimport { logger } from '../utils/logger.js';\nimport { ElementType } from './types.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\n\n// File operation constants\nexport const FILE_CONSTANTS = {\n  ELEMENT_EXTENSION: '.md',\n  YAML_EXTENSION: '.yaml',\n  YML_EXTENSION: '.yml',\n  JSON_EXTENSION: '.json',\n  MAX_FILE_SIZE: 10 * 1024 * 1024, // 10MB max file size for safety\n  CHECKSUM_ALGORITHM: 'sha256',\n  FILE_PERMISSIONS: 0o644,\n  CHUNK_SIZE: 64 * 1024 // 64KB chunks for reading large files\n} as const;\n\n// Internal constants\nconst DATA_DIR_CACHE_KEY = 'dollhouse_data_dir';\nconst COPY_RETRY_ATTEMPTS = 3;\nconst COPY_RETRY_DELAY = 100; // ms\n\nexport interface DefaultElementProviderConfig {\n  /** Custom data directory paths to search (checked before default paths) */\n  customDataPaths?: string[];\n  /** Whether to use default search paths after custom paths */\n  useDefaultPaths?: boolean;\n}\n\nexport class DefaultElementProvider {\n  private readonly __dirname: string;\n  private static cachedDataDir: string | null = null;\n  private static populateInProgress: Map<string, Promise<void>> = new Map();\n  private readonly config: DefaultElementProviderConfig;\n  \n  constructor(config?: DefaultElementProviderConfig) {\n    const __filename = fileURLToPath(import.meta.url);\n    this.__dirname = path.dirname(__filename);\n    this.config = {\n      useDefaultPaths: true,\n      ...config\n    };\n  }\n  \n  /**\n   * Search paths for bundled data directory\n   * Ordered by priority - custom paths first, then development/git, then NPM locations\n   */\n  private get dataSearchPaths(): string[] {\n    const paths: string[] = [];\n    \n    // Add custom paths first (highest priority)\n    if (this.config.customDataPaths) {\n      paths.push(...this.config.customDataPaths);\n    }\n    \n    // Add default paths if enabled\n    if (this.config.useDefaultPaths !== false) {\n      paths.push(\n        // Development/Git installation (relative to this file)\n        path.join(this.__dirname, '../../data'),\n        path.join(this.__dirname, '../../../data'),\n        \n        // NPM installations - macOS Homebrew\n        '/opt/homebrew/lib/node_modules/@dollhousemcp/mcp-server/data',\n        \n        // NPM installations - standard Unix/Linux\n        '/usr/local/lib/node_modules/@dollhousemcp/mcp-server/data',\n        '/usr/lib/node_modules/@dollhousemcp/mcp-server/data',\n        \n        // NPM installations - Windows\n        'C:\\\\Program Files\\\\nodejs\\\\node_modules\\\\@dollhousemcp\\\\mcp-server\\\\data',\n        'C:\\\\Program Files (x86)\\\\nodejs\\\\node_modules\\\\@dollhousemcp\\\\mcp-server\\\\data',\n        \n        // NPM installations - Windows with nvm\n        path.join(process.env.APPDATA || '', 'npm', 'node_modules', '@dollhousemcp', 'mcp-server', 'data'),\n        \n        // Current working directory (last resort)\n        path.join(process.cwd(), 'data')\n      );\n    }\n    \n    return paths;\n  }\n  \n  /**\n   * Find the bundled data directory by checking each search path\n   * Uses Promise.allSettled for better performance and caches the result\n   */\n  private async findDataDirectory(): Promise<string | null> {\n    // Return cached value if available\n    if (DefaultElementProvider.cachedDataDir !== null) {\n      return DefaultElementProvider.cachedDataDir;\n    }\n    \n    // Check all paths in parallel for better performance\n    const checkPromises = this.dataSearchPaths.map(async (searchPath) => {\n      try {\n        const stats = await fs.stat(searchPath);\n        if (stats.isDirectory()) {\n          // Verify it contains expected subdirectories\n          const hasPersonas = await this.directoryExists(path.join(searchPath, 'personas'));\n          const hasSkills = await this.directoryExists(path.join(searchPath, 'skills'));\n          if (hasPersonas || hasSkills) {\n            return searchPath;\n          }\n        }\n      } catch (error) {\n        // Directory doesn't exist or can't be accessed\n        return null;\n      }\n      return null;\n    });\n    \n    const results = await Promise.allSettled(checkPromises);\n    \n    // Find the first successful result\n    for (const result of results) {\n      if (result.status === 'fulfilled' && result.value !== null) {\n        logger.info(`[DefaultElementProvider] Found data directory at: ${result.value}`);\n        // Cache the result\n        DefaultElementProvider.cachedDataDir = result.value;\n        return result.value;\n      }\n    }\n    \n    logger.warn('[DefaultElementProvider] No bundled data directory found in any search path');\n    return null;\n  }\n  \n  /**\n   * Helper to check if a directory exists\n   */\n  private async directoryExists(dirPath: string): Promise<boolean> {\n    try {\n      const stats = await fs.stat(dirPath);\n      return stats.isDirectory();\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Copy all files from source directory to destination directory\n   * Skips files that already exist to preserve user modifications\n   */\n  private async copyElementFiles(sourceDir: string, destDir: string, elementType: string): Promise<number> {\n    let copiedCount = 0;\n    \n    try {\n      // Ensure destination directory exists\n      await fs.mkdir(destDir, { recursive: true });\n      \n      // Read source directory\n      const files = await fs.readdir(sourceDir);\n      \n      for (const file of files) {\n        // Only copy markdown files\n        if (!file.endsWith(FILE_CONSTANTS.ELEMENT_EXTENSION)) {\n          continue;\n        }\n        \n        // Normalize filename for security\n        const normalizedFile = UnicodeValidator.normalize(file);\n        if (!normalizedFile.isValid) {\n          logger.warn(`[DefaultElementProvider] Skipping file with invalid Unicode: ${file}`);\n          continue;\n        }\n        \n        const sourcePath = path.join(sourceDir, normalizedFile.normalizedContent);\n        const destPath = path.join(destDir, normalizedFile.normalizedContent);\n        \n        try {\n          // Check if destination file already exists\n          await fs.access(destPath);\n          logger.debug(`[DefaultElementProvider] Skipping existing file: ${normalizedFile.normalizedContent}`);\n          continue;\n        } catch {\n          // File doesn't exist, proceed with copy\n        }\n        \n        try {\n          // Validate source file before copying\n          const sourceStats = await fs.stat(sourcePath);\n          \n          // Check file size limit\n          if (sourceStats.size > FILE_CONSTANTS.MAX_FILE_SIZE) {\n            logger.warn(\n              `[DefaultElementProvider] Skipping oversized file ${normalizedFile.normalizedContent}: ` +\n              `${sourceStats.size} bytes (max: ${FILE_CONSTANTS.MAX_FILE_SIZE} bytes)`,\n              { \n                file: normalizedFile.normalizedContent, \n                size: sourceStats.size,\n                maxSize: FILE_CONSTANTS.MAX_FILE_SIZE,\n                elementType \n              }\n            );\n            continue;\n          }\n          \n          // Copy the file with verification\n          await this.copyFileWithVerification(sourcePath, destPath);\n          copiedCount++;\n          logger.debug(`[DefaultElementProvider] Copied ${elementType}: ${normalizedFile.normalizedContent}`);\n          \n          // Log security event for each file copied\n          SecurityMonitor.logSecurityEvent({\n            type: 'FILE_COPIED',\n            severity: 'LOW',\n            source: 'DefaultElementProvider.copyElementFiles',\n            details: `Copied default ${elementType} file: ${normalizedFile.normalizedContent}`,\n            metadata: {\n              sourcePath,\n              destPath,\n              elementType,\n              fileSize: (await fs.stat(destPath)).size\n            }\n          });\n        } catch (error) {\n          const err = error as Error;\n          logger.error(\n            `[DefaultElementProvider] Failed to copy ${normalizedFile.normalizedContent}`,\n            { \n              error: err.message,\n              stack: err.stack,\n              sourcePath,\n              destPath,\n              elementType\n            }\n          );\n          // Continue with other files instead of failing completely\n        }\n      }\n      \n      if (copiedCount > 0) {\n        logger.info(`[DefaultElementProvider] Copied ${copiedCount} ${elementType} file(s)`);\n      }\n      \n    } catch (error) {\n      logger.error(`[DefaultElementProvider] Error copying ${elementType} files:`, error);\n    }\n    \n    return copiedCount;\n  }\n  \n  /**\n   * Copy a file with integrity verification\n   * Ensures the file was copied correctly by comparing sizes\n   */\n  /**\n   * Calculate checksum of a file for integrity verification\n   * @param filePath Path to the file\n   * @returns Hex-encoded checksum\n   */\n  private async calculateChecksum(filePath: string): Promise<string> {\n    const hash = createHash(FILE_CONSTANTS.CHECKSUM_ALGORITHM);\n    const stream = await fs.open(filePath, 'r');\n    \n    try {\n      const buffer = Buffer.alloc(FILE_CONSTANTS.CHUNK_SIZE);\n      let bytesRead: number;\n      \n      do {\n        const result = await stream.read(buffer, 0, FILE_CONSTANTS.CHUNK_SIZE);\n        bytesRead = result.bytesRead;\n        \n        if (bytesRead > 0) {\n          hash.update(buffer.subarray(0, bytesRead));\n        }\n      } while (bytesRead > 0);\n      \n      return hash.digest('hex');\n    } finally {\n      await stream.close();\n    }\n  }\n\n  /**\n   * Copy file with integrity verification and retry logic\n   * @param sourcePath Source file path\n   * @param destPath Destination file path\n   * @throws Error if copy fails after all retry attempts\n   */\n  private async copyFileWithVerification(sourcePath: string, destPath: string): Promise<void> {\n    let lastError: Error | null = null;\n    \n    for (let attempt = 1; attempt <= COPY_RETRY_ATTEMPTS; attempt++) {\n      try {\n        // Copy the file\n        await fs.copyFile(sourcePath, destPath);\n        \n        // Verify size matches\n        const [sourceStats, destStats] = await Promise.all([\n          fs.stat(sourcePath),\n          fs.stat(destPath)\n        ]);\n        \n        if (sourceStats.size !== destStats.size) {\n          throw new Error(\n            `Size mismatch after copy - source: ${sourceStats.size} bytes, ` +\n            `destination: ${destStats.size} bytes`\n          );\n        }\n        \n        // Verify checksum matches for complete integrity\n        const [sourceChecksum, destChecksum] = await Promise.all([\n          this.calculateChecksum(sourcePath),\n          this.calculateChecksum(destPath)\n        ]);\n        \n        if (sourceChecksum !== destChecksum) {\n          throw new Error(\n            `Checksum mismatch after copy - source: ${sourceChecksum}, ` +\n            `destination: ${destChecksum}`\n          );\n        }\n        \n        // Set proper permissions\n        try {\n          await fs.chmod(destPath, FILE_CONSTANTS.FILE_PERMISSIONS);\n        } catch (error) {\n          logger.debug(\n            `[DefaultElementProvider] Could not set permissions on ${destPath}: ${error}`,\n            { sourcePath, destPath, attempt }\n          );\n        }\n        \n        // Success - file copied and verified\n        logger.debug(\n          `[DefaultElementProvider] Successfully copied and verified: ${path.basename(sourcePath)}`,\n          { size: sourceStats.size, checksum: sourceChecksum.substring(0, 8) }\n        );\n        return;\n        \n      } catch (error) {\n        lastError = error as Error;\n        \n        // Clean up failed copy\n        try {\n          await fs.unlink(destPath);\n        } catch {\n          // Ignore cleanup errors\n        }\n        \n        if (attempt < COPY_RETRY_ATTEMPTS) {\n          logger.debug(\n            `[DefaultElementProvider] Copy attempt ${attempt} failed, retrying...`,\n            { error: lastError.message, sourcePath, destPath }\n          );\n          await new Promise(resolve => setTimeout(resolve, COPY_RETRY_DELAY * attempt));\n        }\n      }\n    }\n    \n    // All attempts failed\n    throw new Error(\n      `Failed to copy ${path.basename(sourcePath)} after ${COPY_RETRY_ATTEMPTS} attempts: ` +\n      `${lastError?.message || 'Unknown error'}`\n    );\n  }\n  \n  /**\n   * Populate the portfolio with default elements from bundled data\n   * This is called during portfolio initialization for new installations\n   * Protected against concurrent calls to prevent race conditions\n   */\n  public async populateDefaults(portfolioBaseDir: string): Promise<void> {\n    // Check if population is already in progress for this portfolio\n    const existingPopulation = DefaultElementProvider.populateInProgress.get(portfolioBaseDir);\n    if (existingPopulation) {\n      logger.debug(\n        '[DefaultElementProvider] Population already in progress for portfolio, waiting...',\n        { portfolioBaseDir }\n      );\n      return existingPopulation;\n    }\n    \n    // Create new population promise\n    const populationPromise = this.performPopulation(portfolioBaseDir)\n      .finally(() => {\n        // Clean up when done\n        DefaultElementProvider.populateInProgress.delete(portfolioBaseDir);\n      });\n    \n    DefaultElementProvider.populateInProgress.set(portfolioBaseDir, populationPromise);\n    return populationPromise;\n  }\n  \n  /**\n   * Perform the actual population of default elements\n   * @param portfolioBaseDir Base directory of the portfolio\n   */\n  private async performPopulation(portfolioBaseDir: string): Promise<void> {\n    logger.info(\n      '[DefaultElementProvider] Starting default element population',\n      { portfolioBaseDir }\n    );\n    \n    // Log security event for portfolio initialization\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_INITIALIZATION',\n      severity: 'LOW',\n      source: 'DefaultElementProvider.performPopulation',\n      details: `Starting default element population for portfolio: ${portfolioBaseDir}`\n    });\n    \n    // Find the bundled data directory\n    const dataDir = await this.findDataDirectory();\n    if (!dataDir) {\n      logger.warn(\n        '[DefaultElementProvider] No bundled data directory found - portfolio will start empty',\n        { \n          searchPaths: this.dataSearchPaths.slice(0, 3), // Log first few paths for debugging\n          cwd: process.cwd(),\n          dirname: this.__dirname\n        }\n      );\n      return;\n    }\n    \n    // Track total files copied\n    let totalCopied = 0;\n    const copiedCounts: Record<string, number> = {};\n    \n    // Copy each element type - directories now match enum values (all plural)\n    for (const elementType of Object.values(ElementType)) {\n      const sourceDir = path.join(dataDir, elementType);\n      const destDir = path.join(portfolioBaseDir, elementType);\n      \n      try {\n        // Check if source directory exists\n        await fs.access(sourceDir);\n        const copiedCount = await this.copyElementFiles(sourceDir, destDir, elementType);\n        copiedCounts[elementType] = copiedCount;\n        totalCopied += copiedCount;\n      } catch (error) {\n        // Source directory doesn't exist, skip\n        logger.debug(`[DefaultElementProvider] No ${elementType} directory in bundled data`);\n      }\n    }\n    \n    if (totalCopied > 0) {\n      logger.info(\n        `[DefaultElementProvider] Successfully populated portfolio with ${totalCopied} default element(s)`,\n        {\n          portfolioBaseDir,\n          dataDir,\n          breakdown: copiedCounts\n        }\n      );\n      \n      // Log security event for successful population\n      SecurityMonitor.logSecurityEvent({\n        type: 'PORTFOLIO_POPULATED',\n        severity: 'LOW',\n        source: 'DefaultElementProvider.performPopulation',\n        details: `Successfully populated portfolio with ${totalCopied} default elements`,\n        metadata: {\n          portfolioBaseDir,\n          dataDir,\n          copiedCounts\n        }\n      });\n    } else {\n      logger.info('[DefaultElementProvider] No new elements to copy - portfolio may already have content');\n    }\n  }\n}"]}
|
|
878
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DefaultElementProvider.js","sourceRoot":"","sources":["../../src/portfolio/DefaultElementProvider.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * DefaultElementProvider - Populates portfolio with default elements from bundled data\n * \n * This class handles copying default personas, skills, templates, and other elements\n * from the NPM package or Git repository to the user's portfolio on first run.\n * It ensures users have example content to work with immediately after installation.\n */\n\nimport * as fs from 'fs/promises';\nimport * as fsSync from 'fs';\nimport * as path from 'path';\nimport { fileURLToPath } from 'url';\nimport { createHash } from 'crypto';\nimport * as yaml from 'js-yaml';\nimport { logger } from '../utils/logger.js';\nimport { ElementType } from './types.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\n\n// File operation constants\nexport const FILE_CONSTANTS = {\n  ELEMENT_EXTENSION: '.md',\n  YAML_EXTENSION: '.yaml',\n  YML_EXTENSION: '.yml',\n  JSON_EXTENSION: '.json',\n  MAX_FILE_SIZE: 10 * 1024 * 1024, // 10MB max file size for safety\n  CHECKSUM_ALGORITHM: 'sha256',\n  FILE_PERMISSIONS: 0o644,\n  CHUNK_SIZE: 64 * 1024 // 64KB chunks for reading large files\n} as const;\n\n// Development mode detection\n// When running from a git clone, we don't want to auto-load test data\nconst IS_DEVELOPMENT_MODE = (() => {\n  try {\n    // Check if we're in a git repository (development mode)\n    const gitDir = path.join(process.cwd(), '.git');\n    return fsSync.existsSync(gitDir);\n  } catch {\n    return false;\n  }\n})();\n\n// Internal constants\nconst DATA_DIR_CACHE_KEY = 'dollhouse_data_dir';\nconst COPY_RETRY_ATTEMPTS = 3;\nconst COPY_RETRY_DELAY = 100; // ms\n\nexport interface DefaultElementProviderConfig {\n  /** Custom data directory paths to search (checked before default paths) */\n  customDataPaths?: string[];\n  /** Whether to use default search paths after custom paths */\n  useDefaultPaths?: boolean;\n  /** Whether to load test/example data from repository (default: false in dev mode) */\n  loadTestData?: boolean;\n}\n\n// PERFORMANCE: Metadata cache with mtime-based invalidation\ninterface MetadataCacheEntry {\n  metadata: any;\n  mtime: number;\n  size: number;\n}\n\nexport class DefaultElementProvider {\n  private readonly __dirname: string;\n  private static cachedDataDir: string | null = null;\n  private static populateInProgress: Map<string, Promise<void>> = new Map();\n  private readonly config: DefaultElementProviderConfig;\n  \n  // PERFORMANCE OPTIMIZATION: Cache metadata with file mtime for invalidation\n  private static metadataCache: Map<string, MetadataCacheEntry> = new Map();\n  private static readonly MAX_CACHE_SIZE = 20; // MEMORY LEAK FIX: Further reduced cache size to prevent accumulation during performance tests\n  \n  constructor(config?: DefaultElementProviderConfig) {\n    const __filename = fileURLToPath(import.meta.url);\n    this.__dirname = path.dirname(__filename);\n    \n    // Check environment variable for test data loading\n    const envLoadTestData = process.env.DOLLHOUSE_LOAD_TEST_DATA;\n    const loadTestDataFromEnv = envLoadTestData === 'true' || envLoadTestData === '1';\n    const disableTestDataFromEnv = envLoadTestData === 'false' || envLoadTestData === '0';\n    \n    // Check if we're in development mode (with respect to FORCE_PRODUCTION_MODE override)\n    const isDevMode = (() => {\n      // Respect FORCE_PRODUCTION_MODE override first\n      if (process.env.FORCE_PRODUCTION_MODE === 'true') {\n        return false; // Force production mode\n      }\n      if (process.env.FORCE_PRODUCTION_MODE === 'false') {\n        return true; // Force development mode\n      }\n      // Fall back to git detection\n      return IS_DEVELOPMENT_MODE;\n    })();\n    \n    // Determine loadTestData value\n    let computedLoadTestData: boolean;\n    if (loadTestDataFromEnv) {\n      // Environment explicitly enables test data\n      computedLoadTestData = true;\n    } else if (disableTestDataFromEnv) {\n      // Environment explicitly disables test data\n      computedLoadTestData = false;\n    } else {\n      // Default logic: enable in production, disable in development unless config overrides\n      computedLoadTestData = !isDevMode && (config?.loadTestData ?? true);\n    }\n    \n    this.config = {\n      useDefaultPaths: true,\n      ...config,\n      // Apply final loadTestData logic - environment variables and development mode take precedence\n      loadTestData: loadTestDataFromEnv ? true : disableTestDataFromEnv ? false : computedLoadTestData\n    };\n    \n    if (isDevMode && !this.config.loadTestData) {\n      logger.info('[DefaultElementProvider] Development mode detected - test data loading disabled');\n      logger.info('[DefaultElementProvider] To enable test data, set DOLLHOUSE_LOAD_TEST_DATA=true');\n    }\n  }\n  \n  /**\n   * Get the current loadTestData configuration value\n   * @returns Whether test data loading is enabled\n   */\n  public get isTestDataLoadingEnabled(): boolean {\n    return this.config.loadTestData ?? false;\n  }\n  \n  /**\n   * Get whether the system is in development mode\n   * @returns Whether running in development mode\n   */\n  public get isDevelopmentMode(): boolean {\n    return IS_DEVELOPMENT_MODE;\n  }\n  \n  /**\n   * Search paths for bundled data directory\n   * Ordered by priority - custom paths first, then development/git, then NPM locations\n   */\n  private get dataSearchPaths(): string[] {\n    const paths: string[] = [];\n    \n    // Add custom paths first (highest priority)\n    if (this.config.customDataPaths) {\n      paths.push(...this.config.customDataPaths);\n    }\n    \n    // Add default paths if enabled\n    if (this.config.useDefaultPaths !== false) {\n      // Skip development/repository data paths unless test data loading is enabled\n      if (this.config.loadTestData) {\n        // Development/Git installation (relative to this file)\n        paths.push(\n          path.join(this.__dirname, '../../data'),\n          path.join(this.__dirname, '../../../data'),\n          // Current working directory (last resort)\n          path.join(process.cwd(), 'data')\n        );\n      }\n      \n      // Always include NPM installation paths (these would have production data)\n      paths.push(\n        // NPM installations - macOS Homebrew\n        '/opt/homebrew/lib/node_modules/@dollhousemcp/mcp-server/data',\n        \n        // NPM installations - standard Unix/Linux\n        '/usr/local/lib/node_modules/@dollhousemcp/mcp-server/data',\n        '/usr/lib/node_modules/@dollhousemcp/mcp-server/data',\n        \n        // NPM installations - Windows\n        'C:\\\\Program Files\\\\nodejs\\\\node_modules\\\\@dollhousemcp\\\\mcp-server\\\\data',\n        'C:\\\\Program Files (x86)\\\\nodejs\\\\node_modules\\\\@dollhousemcp\\\\mcp-server\\\\data',\n        \n        // NPM installations - Windows with nvm\n        path.join(process.env.APPDATA || '', 'npm', 'node_modules', '@dollhousemcp', 'mcp-server', 'data')\n      );\n    }\n    \n    return paths;\n  }\n  \n  /**\n   * Find the bundled data directory by checking each search path\n   * Uses Promise.allSettled for better performance and caches the result\n   */\n  private async findDataDirectory(): Promise<string | null> {\n    // Return cached value if available\n    if (DefaultElementProvider.cachedDataDir !== null) {\n      return DefaultElementProvider.cachedDataDir;\n    }\n    \n    // Check all paths in parallel for better performance\n    const checkPromises = this.dataSearchPaths.map(async (searchPath) => {\n      try {\n        const stats = await fs.stat(searchPath);\n        if (stats.isDirectory()) {\n          // Verify it contains expected subdirectories\n          const hasPersonas = await this.directoryExists(path.join(searchPath, 'personas'));\n          const hasSkills = await this.directoryExists(path.join(searchPath, 'skills'));\n          if (hasPersonas || hasSkills) {\n            return searchPath;\n          }\n        }\n      } catch (error) {\n        // Directory doesn't exist or can't be accessed\n        return null;\n      }\n      return null;\n    });\n    \n    const results = await Promise.allSettled(checkPromises);\n    \n    // Find the first successful result\n    for (const result of results) {\n      if (result.status === 'fulfilled' && result.value !== null) {\n        logger.info(`[DefaultElementProvider] Found data directory at: ${result.value}`);\n        // Cache the result\n        DefaultElementProvider.cachedDataDir = result.value;\n        return result.value;\n      }\n    }\n    \n    logger.warn('[DefaultElementProvider] No bundled data directory found in any search path');\n    return null;\n  }\n  \n  /**\n   * Helper to check if a directory exists\n   */\n  private async directoryExists(dirPath: string): Promise<boolean> {\n    try {\n      const stats = await fs.stat(dirPath);\n      return stats.isDirectory();\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Validate file path to prevent path traversal attacks\n   * SECURITY FIX: Added file path validation to prevent directory traversal\n   * Previously: File paths were used without validation, allowing potential ../../../ attacks\n   * Now: Strict validation ensures paths stay within allowed directories\n   * @param filePath The file path to validate\n   * @param allowedBasePaths Array of allowed base paths (optional)\n   * @returns true if path is safe, false otherwise\n   */\n  private validateFilePath(filePath: string, allowedBasePaths?: string[]): boolean {\n    try {\n      // SECURITY: Normalize path to prevent traversal attempts\n      const normalizedPath = path.normalize(filePath);\n      \n      // SECURITY: Reject paths containing traversal patterns\n      if (normalizedPath.includes('..')) {\n        logger.warn(`[DefaultElementProvider] Path traversal attempt blocked: ${filePath}`);\n        return false;\n      }\n      \n      // SECURITY: Check for home directory expansion attempts, but allow Windows 8.3 short path names\n      // Windows short path names use ~ followed by a digit (e.g., RUNNER~1), which should be allowed\n      // Only block ~ followed by / or \\ (home directory expansion patterns)\n      if (normalizedPath.includes('~/') || normalizedPath.includes('~\\\\')) {\n        logger.warn(`[DefaultElementProvider] Home directory expansion attempt blocked: ${filePath}`);\n        return false;\n      }\n      \n      // SECURITY: Reject absolute paths outside allowed directories\n      if (path.isAbsolute(normalizedPath) && allowedBasePaths) {\n        const isAllowed = allowedBasePaths.some(basePath => {\n          const normalizedBase = path.normalize(basePath);\n          return normalizedPath.startsWith(normalizedBase);\n        });\n        \n        if (!isAllowed) {\n          logger.warn(`[DefaultElementProvider] Absolute path outside allowed directories: ${filePath}`);\n          return false;\n        }\n      }\n      \n      // SECURITY: Reject null bytes and other dangerous characters\n      if (normalizedPath.includes('\\0') || normalizedPath.includes('\\x00')) {\n        logger.warn(`[DefaultElementProvider] Null byte in path blocked: ${filePath}`);\n        return false;\n      }\n      \n      return true;\n    } catch (error) {\n      logger.warn(`[DefaultElementProvider] Path validation error: ${error}`);\n      return false;\n    }\n  }\n\n  // DEPRECATED: Commented out filename pattern detection - replaced with metadata-based detection\n  /**\n   * Cached compiled regex patterns for performance optimization\n   * @deprecated Use metadata-based detection instead\n   */\n  // private static compiledTestPatterns: RegExp[] | null = null;\n\n  /**\n   * Get compiled test patterns with caching for better performance\n   * @deprecated Use metadata-based detection instead\n   * @returns Array of compiled regex patterns\n   */\n  // private getCompiledTestPatterns(): RegExp[] {\n  //   // Use cached patterns if available\n  //   if (DefaultElementProvider.compiledTestPatterns) {\n  //     return DefaultElementProvider.compiledTestPatterns;\n  //   }\n  //   \n  //   // Compile and cache patterns on first use\n  //   // CRITICAL FIX: Removed overly broad /^test-/i pattern that was blocking legitimate use\n  //   // Users should be able to create personas like \"test-driven-developer\" or \"test-automation-expert\"\n  //   // We only block specific test patterns that are clearly from our test suite\n  //   DefaultElementProvider.compiledTestPatterns = [\n  //     /^testpersona/i,              // Our test suite pattern\n  //     /^yamltest/i,                 // Security test pattern\n  //     /^yamlbomb/i,                 // Security test pattern\n  //     /^memory-test-/i,             // Performance test pattern\n  //     /^perf-test-/i,               // Performance test pattern\n  //     /^test-fixture-/i,            // Test fixture pattern (more specific)\n  //     /^test-data-/i,               // Test data pattern (more specific)\n  //     /bin-sh|rm-rf|pwned/i,        // Malicious patterns\n  //     /concurrent-\\d+/i,            // Concurrent test pattern\n  //     /legacy\\.md$/i,               // Legacy test pattern\n  //     /performance-test/i,          // Performance test pattern\n  //     /-\\d{13}-[a-z0-9]+\\.md$/i,    // Timestamp-based test files\n  //     /^unittest-/i,                // Unit test pattern\n  //     /^integrationtest-/i,         // Integration test pattern\n  //   ];\n  //   \n  //   return DefaultElementProvider.compiledTestPatterns;\n  // }\n\n  /**\n   * Check if a filename matches test data patterns that should never be copied to production\n   * @deprecated Use isDollhouseMCPTestElement() for metadata-based detection instead\n   * @param filename The filename to check\n   * @returns true if the filename matches test patterns that should be blocked\n   */\n  // private isTestDataPattern(filename: string): boolean {\n  //   const patterns = this.getCompiledTestPatterns();\n  //   return patterns.some(pattern => pattern.test(filename));\n  // }\n\n  /**\n   * Read metadata from YAML frontmatter only (never reads content body)\n   * Uses a small buffer to safely extract only the frontmatter between --- markers\n   * @param filePath Path to the file to read metadata from\n   * @returns Parsed metadata object or null if no frontmatter found\n   */\n  // PERFORMANCE OPTIMIZATION: Reusable buffer pool to reduce allocations\n  private static readonly bufferPool: Buffer[] = [];\n  private static readonly MAX_POOL_SIZE = 20; // MEMORY LEAK FIX: Reduced buffer pool size to match cache size for consistent memory management\n  private static bufferPoolStats = { hits: 0, misses: 0, created: 0 };\n  \n  private getBuffer(): Buffer {\n    // PERFORMANCE: Track buffer pool usage for optimization monitoring\n    let buffer = DefaultElementProvider.bufferPool.pop();\n    if (buffer) {\n      DefaultElementProvider.bufferPoolStats.hits++;\n      return buffer;\n    } else {\n      DefaultElementProvider.bufferPoolStats.misses++;\n      DefaultElementProvider.bufferPoolStats.created++;\n      buffer = Buffer.alloc(4096);\n      logger.debug(`[DefaultElementProvider] Created new buffer (pool empty), total created: ${DefaultElementProvider.bufferPoolStats.created}`);\n      return buffer;\n    }\n  }\n  \n  private releaseBuffer(buffer: Buffer): void {\n    // CRITICAL FIX: Always attempt to return buffer to pool for reuse\n    if (DefaultElementProvider.bufferPool.length < DefaultElementProvider.MAX_POOL_SIZE) {\n      buffer.fill(0); // SECURITY: Clear buffer before reuse to prevent data leakage\n      DefaultElementProvider.bufferPool.push(buffer);\n    } else {\n      // PERFORMANCE: If pool is full, clear the buffer to help GC\n      buffer.fill(0);\n      logger.debug('[DefaultElementProvider] Buffer pool full, discarding buffer');\n    }\n  }\n\n  /**\n   * Clean up buffer pool and cache to free memory\n   * PERFORMANCE FIX: Added cleanup method to prevent memory leaks\n   * This should be called during application shutdown or periodic cleanup\n   */\n  public static cleanup(): void {\n    // Clear buffer pool\n    DefaultElementProvider.bufferPool.length = 0;\n    \n    // Clear metadata cache\n    DefaultElementProvider.metadataCache.clear();\n    \n    // Clear cached data directory\n    DefaultElementProvider.cachedDataDir = null;\n    \n    // Clear population promises\n    DefaultElementProvider.populateInProgress.clear();\n    \n    logger.info('[DefaultElementProvider] Memory cleanup completed', {\n      bufferStats: DefaultElementProvider.bufferPoolStats,\n      cacheCleared: true\n    });\n    \n    // Reset stats\n    DefaultElementProvider.bufferPoolStats = { hits: 0, misses: 0, created: 0 };\n  }\n\n  /**\n   * Get performance statistics for monitoring\n   * PERFORMANCE MONITORING: Added statistics method for performance tracking\n   * This provides insights into buffer pool efficiency and cache performance\n   * @returns Object containing performance metrics\n   */\n  public static getPerformanceStats(): {\n    bufferPool: {\n      hits: number;\n      misses: number; \n      created: number;\n      hitRate: number;\n      poolSize: number;\n      maxPoolSize: number;\n    };\n    metadataCache: {\n      size: number;\n      maxSize: number;\n    };\n  } {\n    const bufferHits = DefaultElementProvider.bufferPoolStats.hits;\n    const bufferMisses = DefaultElementProvider.bufferPoolStats.misses;\n    const totalRequests = bufferHits + bufferMisses;\n    \n    return {\n      bufferPool: {\n        hits: bufferHits,\n        misses: bufferMisses,\n        created: DefaultElementProvider.bufferPoolStats.created,\n        hitRate: totalRequests > 0 ? bufferHits / totalRequests : 0,\n        poolSize: DefaultElementProvider.bufferPool.length,\n        maxPoolSize: DefaultElementProvider.MAX_POOL_SIZE\n      },\n      metadataCache: {\n        size: DefaultElementProvider.metadataCache.size,\n        maxSize: DefaultElementProvider.MAX_CACHE_SIZE\n      }\n    };\n  }\n\n  private async readMetadataOnly(filePath: string, retries = 2): Promise<any | null> {\n    // PERFORMANCE: Check cache first before reading file\n    try {\n      const stats = await fs.stat(filePath);\n      const cacheKey = filePath;\n      const cached = DefaultElementProvider.metadataCache.get(cacheKey);\n      \n      // Return cached metadata if file hasn't changed \n      // CRITICAL FIX: Use integer mtime comparison to avoid floating-point precision issues\n      if (cached && Math.floor(cached.mtime) === Math.floor(stats.mtimeMs) && cached.size === stats.size) {\n        logger.debug(`[DefaultElementProvider] Cache hit for ${filePath}`);\n        return cached.metadata;\n      }\n    } catch {\n      // File doesn't exist, proceed with normal flow\n    }\n    \n    try {\n      // Open file and read only first 4KB to avoid reading dangerous content\n      const fd = await fs.open(filePath, 'r');\n      // PERFORMANCE: Use buffer pool instead of allocating new buffer each time\n      const buffer = this.getBuffer();\n      \n      try {\n        const result = await fd.read(buffer, 0, 4096, 0);\n        const header = buffer.subarray(0, result.bytesRead).toString('utf-8');\n        \n        // Look for YAML frontmatter between --- markers\n        // Support both Unix (\\n) and Windows (\\r\\n) line endings\n        const match = header.match(/^---\\r?\\n([\\s\\S]*?)\\r?\\n---/);\n        if (!match) {\n          return null; // No frontmatter found\n        }\n        \n        // Parse the YAML frontmatter safely\n        try {\n          // SECURITY FIX: Replace direct YAML parsing function with SecureYamlParser for enhanced security\n          // SecureYamlParser provides additional validation, injection prevention, and content sanitization\n          // It expects full YAML with --- markers, so we reconstruct the frontmatter block\n          // We disable specific field validation as this is general metadata parsing, not persona-specific\n          const fullYaml = `---\\n${match[1]}\\n---`;\n          const parseResult = SecureYamlParser.parse(fullYaml, { \n            validateContent: false, \n            validateFields: false \n          });\n          const metadata = parseResult.data;\n          \n          // PERFORMANCE: Cache the metadata with file stats for future reads\n          if (typeof metadata === 'object' && metadata !== null) {\n            try {\n              const stats = await fs.stat(filePath);\n              const cacheEntry: MetadataCacheEntry = {\n                metadata,\n                mtime: stats.mtimeMs,\n                size: stats.size\n              };\n              \n              // CRITICAL MEMORY LEAK FIX: More aggressive cache management to prevent unbounded growth\n              // Check if this entry already exists and just update it instead of adding new\n              if (DefaultElementProvider.metadataCache.has(filePath)) {\n                // Update existing entry - no eviction needed\n                DefaultElementProvider.metadataCache.set(filePath, cacheEntry);\n                logger.debug(`[DefaultElementProvider] Updated existing cache entry for ${filePath}`);\n              } else {\n                // New entry - check if we need to evict first\n                // Use > instead of >= to ensure we never exceed MAX_CACHE_SIZE\n                if (DefaultElementProvider.metadataCache.size >= DefaultElementProvider.MAX_CACHE_SIZE) {\n                  // More aggressive eviction: remove enough entries to stay well under limit\n                  const entriesToEvict = Math.max(1, Math.floor(DefaultElementProvider.MAX_CACHE_SIZE * 0.4));\n                  const keysToEvict = Array.from(DefaultElementProvider.metadataCache.keys()).slice(0, entriesToEvict);\n                  for (const key of keysToEvict) {\n                    DefaultElementProvider.metadataCache.delete(key);\n                  }\n                  logger.debug(`[DefaultElementProvider] Evicted ${keysToEvict.length} cache entries to manage memory (cache size was ${DefaultElementProvider.metadataCache.size + keysToEvict.length})`);\n                }\n                \n                DefaultElementProvider.metadataCache.set(filePath, cacheEntry);\n                logger.debug(`[DefaultElementProvider] Added new cache entry for ${filePath} (cache size now: ${DefaultElementProvider.metadataCache.size})`);\n              }\n            } catch {\n              // Ignore cache errors, return metadata anyway\n            }\n            return metadata;\n          }\n          return null;\n        } catch (yamlError) {\n          // Invalid YAML, return null\n          // ENHANCEMENT: Include error type for better debugging\n          const yamlErrorType = (yamlError as any)?.constructor?.name || 'YAMLError';\n          logger.debug(`[DefaultElementProvider] Invalid YAML in ${filePath}: ${yamlErrorType} - ${yamlError}`);\n          return null;\n        }\n      } finally {\n        // CRITICAL FIX: Ensure file descriptor is closed and buffer is released in ALL paths\n        try {\n          await fd.close();\n        } catch (closeError) {\n          logger.debug(`[DefaultElementProvider] Error closing file descriptor for ${filePath}: ${closeError}`);\n        }\n        // PERFORMANCE: Return buffer to pool for reuse\n        this.releaseBuffer(buffer);\n      }\n    } catch (error: any) {\n      // ENHANCEMENT: Include error type in debug logs for better debugging\n      const errorType = error?.constructor?.name || 'UnknownError';\n      const errorCode = error?.code || 'NO_CODE';\n      \n      // RELIABILITY: Add retry logic for transient failures\n      if (retries > 0 && (errorCode === 'EBUSY' || errorCode === 'EAGAIN')) {\n        logger.debug(`[DefaultElementProvider] Retrying read for ${filePath} after ${errorType}:${errorCode}`);\n        await new Promise(resolve => setTimeout(resolve, 50)); // Brief delay before retry\n        return this.readMetadataOnly(filePath, retries - 1);\n      }\n      \n      logger.debug(`[DefaultElementProvider] Could not read metadata from ${filePath}: ${errorType}:${errorCode} - ${error?.message || error}`);\n      return null;\n    }\n  }\n\n  /**\n   * Check if a file is a DollhouseMCP test element based on metadata\n   * This replaces filename pattern detection with accurate metadata-based detection\n   * @param filePath Path to the file to check\n   * @returns true if the file contains _dollhouseMCPTest: true metadata\n   */\n  private async isDollhouseMCPTestElement(filePath: string): Promise<boolean> {\n    try {\n      const metadata = await this.readMetadataOnly(filePath);\n      const isTest = !!(metadata && metadata._dollhouseMCPTest === true);\n      \n      \n      return isTest;\n    } catch (error) {\n      // If we can't read the metadata, assume it's not a test file\n      logger.debug(`[DefaultElementProvider] Error checking test metadata for ${filePath}: ${error}`);\n      return false;\n    }\n  }\n\n  /**\n   * Detect if we're in a production environment by checking for production indicators\n   * Uses a confidence-based approach requiring multiple indicators for better accuracy\n   * @returns true if this appears to be a production environment\n   */\n  private isProductionEnvironment(): boolean {\n    // Allow tests to explicitly override production mode detection\n    if (process.env.FORCE_PRODUCTION_MODE === 'true') {\n      return true;\n    }\n    if (process.env.FORCE_PRODUCTION_MODE === 'false') {\n      return false;\n    }\n    \n    // Weighted indicators for production detection\n    const indicators = {\n      // Strong indicators (weight: 2)\n      hasUserHomeDir: (process.env.HOME && (process.env.HOME.includes('/Users/') || process.env.HOME.includes('/home/'))) || \n                      !!process.env.USERPROFILE,\n      isProductionNode: process.env.NODE_ENV === 'production',\n      notInTestDir: (() => {\n        const cwd = process.cwd().toLowerCase();\n        // Normalize path separators for cross-platform checking (Windows uses \\ but checks use /)\n        const normalizedCwd = cwd.replace(/\\\\/g, '/');\n        return !normalizedCwd.includes('/test') && \n               !normalizedCwd.includes('/__tests__') && \n               !normalizedCwd.includes('/temp') &&\n               !normalizedCwd.includes('/dist/test');\n      })(),\n      \n      // Moderate indicators (weight: 1)\n      notInCI: !process.env.CI,\n      noTestEnv: process.env.NODE_ENV !== 'test',\n      noDevEnv: process.env.NODE_ENV !== 'development',\n    };\n    \n    // Calculate weighted score\n    let score = 0;\n    if (indicators.hasUserHomeDir) score += 2;\n    if (indicators.isProductionNode) score += 2;\n    if (indicators.notInTestDir) score += 2;\n    if (indicators.notInCI) score += 1;\n    if (indicators.noTestEnv) score += 1;\n    if (indicators.noDevEnv) score += 1;\n    \n    // Log detection details for debugging\n    const activeIndicators = Object.entries(indicators)\n      .filter(([_, value]) => value)\n      .map(([key]) => key);\n    \n    // TYPESCRIPT FIX: Removed logger.isDebugEnabled() check as this method doesn't exist on MCPLogger\n    // The logger already handles debug level internally, so we can call debug() directly\n    if (score >= 3) {\n      logger.debug(\n        '[DefaultElementProvider] Production environment detected',\n        { score, activeIndicators, forceMode: 'not set' }\n      );\n    }\n    \n    // Require a score of at least 3 for production detection (more confident)\n    // This prevents false positives in edge cases while maintaining security\n    return score >= 3;\n  }\n  \n  /**\n   * Copy all files from source directory to destination directory\n   * Skips files that already exist to preserve user modifications\n   */\n  private async copyElementFiles(sourceDir: string, destDir: string, elementType: string): Promise<number> {\n    let copiedCount = 0;\n    \n    \n    try {\n      // Ensure destination directory exists\n      await fs.mkdir(destDir, { recursive: true });\n      \n      // Read source directory\n      const files = await fs.readdir(sourceDir);\n      \n      \n      for (const file of files) {\n        \n        // Only copy markdown files\n        if (!file.endsWith(FILE_CONSTANTS.ELEMENT_EXTENSION)) {\n          continue;\n        }\n        \n        // Normalize filename for security\n        const normalizedFile = UnicodeValidator.normalize(file);\n        if (!normalizedFile.isValid) {\n          logger.warn(`[DefaultElementProvider] Skipping file with invalid Unicode: ${file}`);\n          continue;\n        }\n\n        const sourcePath = path.join(sourceDir, normalizedFile.normalizedContent);\n        const destPath = path.join(destDir, normalizedFile.normalizedContent);\n        \n        \n        // SECURITY FIX: Validate file paths to prevent path traversal attacks\n        // This prevents malicious files from escaping the intended directory structure\n        const sourceValid = this.validateFilePath(sourcePath, [sourceDir]);\n        const destValid = this.validateFilePath(destPath, [destDir]);\n        \n        if (!sourceValid || !destValid) {\n          logger.warn(\n            `[DefaultElementProvider] Skipping file with invalid path: ${normalizedFile.normalizedContent}`,\n            { sourcePath, destPath, elementType }\n          );\n          continue;\n        }\n        \n        // Production safety check: Block DollhouseMCP test elements in production environments\n        // Skip this check if loadTestData is explicitly enabled (for testing scenarios)\n        if (!this.config.loadTestData && this.isProductionEnvironment()) {\n          const isDollhouseTest = await this.isDollhouseMCPTestElement(sourcePath);\n          \n          if (isDollhouseTest) {\n            logger.warn(\n              `[DefaultElementProvider] SECURITY: Blocking DollhouseMCP test element in production: ${normalizedFile.normalizedContent}`,\n              { \n                file: normalizedFile.normalizedContent,\n                reason: 'DollhouseMCP test element detected in production environment',\n                elementType\n              }\n            );\n            \n            // Log security event for blocked test data\n            SecurityMonitor.logSecurityEvent({\n              type: 'TEST_DATA_BLOCKED',\n              severity: 'MEDIUM',\n              source: 'DefaultElementProvider.copyElementFiles',\n              details: `Blocked DollhouseMCP test element in production: ${normalizedFile.normalizedContent}`,\n              metadata: {\n                filename: normalizedFile.normalizedContent,\n                elementType,\n                reason: 'DollhouseMCP test element detected in production environment',\n                detectionMethod: 'metadata-based'\n              }\n            });\n            \n            continue;\n          }\n        }\n        \n        try {\n          // Check if destination file already exists\n          await fs.access(destPath);\n          logger.debug(`[DefaultElementProvider] Skipping existing file: ${normalizedFile.normalizedContent}`);\n          continue;\n        } catch {\n          // File doesn't exist, proceed with copy\n        }\n        \n        try {\n          // Validate source file before copying\n          const sourceStats = await fs.stat(sourcePath);\n          \n          // Check file size limit\n          if (sourceStats.size > FILE_CONSTANTS.MAX_FILE_SIZE) {\n            logger.warn(\n              `[DefaultElementProvider] Skipping oversized file ${normalizedFile.normalizedContent}: ` +\n              `${sourceStats.size} bytes (max: ${FILE_CONSTANTS.MAX_FILE_SIZE} bytes)`,\n              { \n                file: normalizedFile.normalizedContent, \n                size: sourceStats.size,\n                maxSize: FILE_CONSTANTS.MAX_FILE_SIZE,\n                elementType \n              }\n            );\n            continue;\n          }\n          \n          // Copy the file with verification\n          \n          await this.copyFileWithVerification(sourcePath, destPath);\n          copiedCount++;\n          \n          logger.debug(`[DefaultElementProvider] Copied ${elementType}: ${normalizedFile.normalizedContent}`);\n          \n          // Log security event for each file copied\n          SecurityMonitor.logSecurityEvent({\n            type: 'FILE_COPIED',\n            severity: 'LOW',\n            source: 'DefaultElementProvider.copyElementFiles',\n            details: `Copied default ${elementType} file: ${normalizedFile.normalizedContent}`,\n            metadata: {\n              sourcePath,\n              destPath,\n              elementType,\n              fileSize: (await fs.stat(destPath)).size\n            }\n          });\n        } catch (error) {\n          const err = error as Error;\n          logger.error(\n            `[DefaultElementProvider] Failed to copy ${normalizedFile.normalizedContent}`,\n            { \n              error: err.message,\n              stack: err.stack,\n              sourcePath,\n              destPath,\n              elementType\n            }\n          );\n          // Continue with other files instead of failing completely\n        }\n      }\n      \n      if (copiedCount > 0) {\n        logger.info(`[DefaultElementProvider] Copied ${copiedCount} ${elementType} file(s)`);\n      }\n      \n    } catch (error) {\n      logger.error(`[DefaultElementProvider] Error copying ${elementType} files:`, error);\n    }\n    \n    return copiedCount;\n  }\n  \n  /**\n   * Copy a file with integrity verification\n   * Ensures the file was copied correctly by comparing sizes\n   */\n  /**\n   * Calculate checksum of a file for integrity verification\n   * @param filePath Path to the file\n   * @returns Hex-encoded checksum\n   */\n  private async calculateChecksum(filePath: string): Promise<string> {\n    const hash = createHash(FILE_CONSTANTS.CHECKSUM_ALGORITHM);\n    const stream = await fs.open(filePath, 'r');\n    \n    try {\n      const buffer = Buffer.alloc(FILE_CONSTANTS.CHUNK_SIZE);\n      let bytesRead: number;\n      \n      do {\n        const result = await stream.read(buffer, 0, FILE_CONSTANTS.CHUNK_SIZE);\n        bytesRead = result.bytesRead;\n        \n        if (bytesRead > 0) {\n          hash.update(buffer.subarray(0, bytesRead));\n        }\n      } while (bytesRead > 0);\n      \n      return hash.digest('hex');\n    } finally {\n      await stream.close();\n    }\n  }\n\n  /**\n   * Copy file with integrity verification and retry logic\n   * @param sourcePath Source file path\n   * @param destPath Destination file path\n   * @throws Error if copy fails after all retry attempts\n   */\n  private async copyFileWithVerification(sourcePath: string, destPath: string): Promise<void> {\n    let lastError: Error | null = null;\n    \n    for (let attempt = 1; attempt <= COPY_RETRY_ATTEMPTS; attempt++) {\n      try {\n        // Copy the file\n        await fs.copyFile(sourcePath, destPath);\n        \n        // Verify size matches\n        const [sourceStats, destStats] = await Promise.all([\n          fs.stat(sourcePath),\n          fs.stat(destPath)\n        ]);\n        \n        if (sourceStats.size !== destStats.size) {\n          throw new Error(\n            `Size mismatch after copy - source: ${sourceStats.size} bytes, ` +\n            `destination: ${destStats.size} bytes`\n          );\n        }\n        \n        // Verify checksum matches for complete integrity\n        const [sourceChecksum, destChecksum] = await Promise.all([\n          this.calculateChecksum(sourcePath),\n          this.calculateChecksum(destPath)\n        ]);\n        \n        if (sourceChecksum !== destChecksum) {\n          throw new Error(\n            `Checksum mismatch after copy - source: ${sourceChecksum}, ` +\n            `destination: ${destChecksum}`\n          );\n        }\n        \n        // Set proper permissions\n        try {\n          await fs.chmod(destPath, FILE_CONSTANTS.FILE_PERMISSIONS);\n        } catch (error) {\n          logger.debug(\n            `[DefaultElementProvider] Could not set permissions on ${destPath}: ${error}`,\n            { sourcePath, destPath, attempt }\n          );\n        }\n        \n        // Success - file copied and verified\n        logger.debug(\n          `[DefaultElementProvider] Successfully copied and verified: ${path.basename(sourcePath)}`,\n          { size: sourceStats.size, checksum: sourceChecksum.substring(0, 8) }\n        );\n        return;\n        \n      } catch (error) {\n        lastError = error as Error;\n        \n        // Clean up failed copy\n        try {\n          await fs.unlink(destPath);\n        } catch {\n          // Ignore cleanup errors\n        }\n        \n        if (attempt < COPY_RETRY_ATTEMPTS) {\n          logger.debug(\n            `[DefaultElementProvider] Copy attempt ${attempt} failed, retrying...`,\n            { error: lastError.message, sourcePath, destPath }\n          );\n          await new Promise(resolve => setTimeout(resolve, COPY_RETRY_DELAY * attempt));\n        }\n      }\n    }\n    \n    // All attempts failed\n    throw new Error(\n      `Failed to copy ${path.basename(sourcePath)} after ${COPY_RETRY_ATTEMPTS} attempts: ` +\n      `${lastError?.message || 'Unknown error'}`\n    );\n  }\n  \n  /**\n   * Populate the portfolio with default elements from bundled data\n   * This is called during portfolio initialization for new installations\n   * Protected against concurrent calls to prevent race conditions\n   */\n  public async populateDefaults(portfolioBaseDir: string): Promise<void> {\n    // Check if population is already in progress for this portfolio\n    const existingPopulation = DefaultElementProvider.populateInProgress.get(portfolioBaseDir);\n    if (existingPopulation) {\n      logger.debug(\n        '[DefaultElementProvider] Population already in progress for portfolio, waiting...',\n        { portfolioBaseDir }\n      );\n      return existingPopulation;\n    }\n    \n    // Create new population promise\n    const populationPromise = this.performPopulation(portfolioBaseDir)\n      .finally(() => {\n        // Clean up when done\n        DefaultElementProvider.populateInProgress.delete(portfolioBaseDir);\n      });\n    \n    DefaultElementProvider.populateInProgress.set(portfolioBaseDir, populationPromise);\n    return populationPromise;\n  }\n  \n  /**\n   * Perform the actual population of default elements\n   * @param portfolioBaseDir Base directory of the portfolio\n   */\n  private async performPopulation(portfolioBaseDir: string): Promise<void> {\n    // Check if test data loading is disabled\n    // Note: This check is needed even though constructor sets config, because\n    // config can be overridden after construction\n    \n    // Use production environment detection that respects FORCE_PRODUCTION_MODE\n    const isDevelopmentMode = !this.isProductionEnvironment();\n    \n    if (isDevelopmentMode && !this.config.loadTestData) {\n      logger.info(\n        '[DefaultElementProvider] Skipping default element population in development mode',\n        { \n          portfolioBaseDir,\n          reason: 'Test data loading disabled',\n          enableWith: 'Set DOLLHOUSE_LOAD_TEST_DATA=true to enable'\n        }\n      );\n      return;\n    }\n    \n    logger.info(\n      '[DefaultElementProvider] Starting default element population',\n      { portfolioBaseDir }\n    );\n    \n    // Log security event for portfolio initialization\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_INITIALIZATION',\n      severity: 'LOW',\n      source: 'DefaultElementProvider.performPopulation',\n      details: `Starting default element population for portfolio: ${portfolioBaseDir}`\n    });\n    \n    // Find the bundled data directory\n    const dataDir = await this.findDataDirectory();\n    if (!dataDir) {\n      logger.warn(\n        '[DefaultElementProvider] No bundled data directory found - portfolio will start empty',\n        { \n          searchPaths: this.dataSearchPaths.slice(0, 3), // Log first few paths for debugging\n          cwd: process.cwd(),\n          dirname: this.__dirname\n        }\n      );\n      return;\n    }\n    \n    // Track total files copied\n    let totalCopied = 0;\n    const copiedCounts: Record<string, number> = {};\n    \n    // Copy each element type - directories now match enum values (all plural)\n    for (const elementType of Object.values(ElementType)) {\n      const sourceDir = path.join(dataDir, elementType);\n      const destDir = path.join(portfolioBaseDir, elementType);\n      \n      try {\n        // Check if source directory exists\n        await fs.access(sourceDir);\n        const copiedCount = await this.copyElementFiles(sourceDir, destDir, elementType);\n        copiedCounts[elementType] = copiedCount;\n        totalCopied += copiedCount;\n      } catch (error) {\n        // Source directory doesn't exist, skip\n        logger.debug(`[DefaultElementProvider] No ${elementType} directory in bundled data`);\n      }\n    }\n    \n    if (totalCopied > 0) {\n      logger.info(\n        `[DefaultElementProvider] Successfully populated portfolio with ${totalCopied} default element(s)`,\n        {\n          portfolioBaseDir,\n          dataDir,\n          breakdown: copiedCounts\n        }\n      );\n      \n      // Log security event for successful population\n      SecurityMonitor.logSecurityEvent({\n        type: 'PORTFOLIO_POPULATED',\n        severity: 'LOW',\n        source: 'DefaultElementProvider.performPopulation',\n        details: `Successfully populated portfolio with ${totalCopied} default elements`,\n        metadata: {\n          portfolioBaseDir,\n          dataDir,\n          copiedCounts\n        }\n      });\n    } else {\n      logger.info('[DefaultElementProvider] No new elements to copy - portfolio may already have content');\n    }\n  }\n}"]}
|