@dollhousemcp/mcp-server 1.5.2 → 1.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +56 -0
- package/README.md +494 -111
- package/data/agents/code-reviewer.md +8 -1
- package/data/agents/research-assistant.md +8 -1
- package/data/agents/task-manager.md +8 -1
- package/data/ensembles/business-advisor.md +8 -1
- package/data/ensembles/creative-studio.md +8 -1
- package/data/ensembles/development-team.md +8 -1
- package/data/ensembles/security-analysis-team.md +8 -1
- package/data/memories/conversation-history.md +8 -1
- package/data/memories/learning-progress.md +8 -1
- package/data/memories/project-context.md +8 -1
- package/data/personas/business-consultant.md +8 -1
- package/data/personas/creative-writer.md +8 -1
- package/data/personas/debug-detective.md +8 -1
- package/data/personas/eli5-explainer.md +8 -1
- package/data/personas/security-analyst.md +8 -1
- package/data/personas/technical-analyst.md +8 -1
- package/data/skills/code-review.md +8 -1
- package/data/skills/creative-writing.md +8 -1
- package/data/skills/data-analysis.md +8 -1
- package/data/skills/penetration-testing.md +8 -1
- package/data/skills/research.md +8 -1
- package/data/skills/threat-modeling.md +8 -1
- package/data/skills/translation.md +8 -1
- package/data/templates/code-documentation.md +8 -1
- package/data/templates/email-professional.md +8 -1
- package/data/templates/meeting-notes.md +8 -1
- package/data/templates/penetration-test-report.md +8 -1
- package/data/templates/project-brief.md +8 -1
- package/data/templates/report-executive.md +8 -1
- package/data/templates/security-vulnerability-report.md +8 -1
- package/data/templates/threat-assessment-report.md +8 -1
- package/dist/auth/GitHubAuthManager.d.ts +6 -1
- package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
- package/dist/auth/GitHubAuthManager.js +45 -18
- package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
- package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
- package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
- package/dist/cache/CollectionCache.d.ts.map +1 -1
- package/dist/cache/CollectionCache.js +13 -3
- package/dist/cache/CollectionIndexCache.d.ts +77 -0
- package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
- package/dist/cache/CollectionIndexCache.js +349 -0
- package/dist/cache/LRUCache.d.ts +93 -0
- package/dist/cache/LRUCache.d.ts.map +1 -0
- package/dist/cache/LRUCache.js +299 -0
- package/dist/cache/index.d.ts +1 -0
- package/dist/cache/index.d.ts.map +1 -1
- package/dist/cache/index.js +2 -1
- package/dist/collection/CollectionBrowser.d.ts +21 -1
- package/dist/collection/CollectionBrowser.d.ts.map +1 -1
- package/dist/collection/CollectionBrowser.js +130 -10
- package/dist/collection/CollectionIndexManager.d.ts +151 -0
- package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
- package/dist/collection/CollectionIndexManager.js +499 -0
- package/dist/collection/CollectionSearch.d.ts +55 -0
- package/dist/collection/CollectionSearch.d.ts.map +1 -1
- package/dist/collection/CollectionSearch.js +338 -13
- package/dist/collection/CollectionSeeder.d.ts.map +1 -1
- package/dist/collection/CollectionSeeder.js +38 -1
- package/dist/collection/ElementInstaller.d.ts +31 -0
- package/dist/collection/ElementInstaller.d.ts.map +1 -1
- package/dist/collection/ElementInstaller.js +77 -15
- package/dist/collection/PersonaSubmitter.d.ts +1 -1
- package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
- package/dist/collection/PersonaSubmitter.js +2 -2
- package/dist/collection/index.d.ts +1 -0
- package/dist/collection/index.d.ts.map +1 -1
- package/dist/collection/index.js +2 -1
- package/dist/config/ConfigManager.d.ts +78 -0
- package/dist/config/ConfigManager.d.ts.map +1 -0
- package/dist/config/ConfigManager.js +216 -0
- package/dist/config/element-types.d.ts +135 -0
- package/dist/config/element-types.d.ts.map +1 -0
- package/dist/config/element-types.js +108 -0
- package/dist/config/index.d.ts +2 -0
- package/dist/config/index.d.ts.map +1 -1
- package/dist/config/index.js +3 -1
- package/dist/config/portfolio-constants.d.ts +83 -0
- package/dist/config/portfolio-constants.d.ts.map +1 -0
- package/dist/config/portfolio-constants.js +99 -0
- package/dist/elements/BaseElement.d.ts +14 -2
- package/dist/elements/BaseElement.d.ts.map +1 -1
- package/dist/elements/BaseElement.js +88 -6
- package/dist/elements/agents/Agent.d.ts +10 -1
- package/dist/elements/agents/Agent.d.ts.map +1 -1
- package/dist/elements/agents/Agent.js +66 -19
- package/dist/elements/agents/AgentManager.d.ts +2 -0
- package/dist/elements/agents/AgentManager.d.ts.map +1 -1
- package/dist/elements/agents/AgentManager.js +12 -10
- package/dist/elements/skills/Skill.d.ts +10 -1
- package/dist/elements/skills/Skill.d.ts.map +1 -1
- package/dist/elements/skills/Skill.js +40 -3
- package/dist/elements/skills/SkillManager.d.ts +1 -0
- package/dist/elements/skills/SkillManager.d.ts.map +1 -1
- package/dist/elements/skills/SkillManager.js +10 -4
- package/dist/elements/templates/Template.d.ts +10 -1
- package/dist/elements/templates/Template.d.ts.map +1 -1
- package/dist/elements/templates/Template.js +35 -18
- package/dist/elements/templates/TemplateManager.d.ts +1 -1
- package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
- package/dist/elements/templates/TemplateManager.js +6 -5
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/index.barrel.d.ts +1 -2
- package/dist/index.barrel.d.ts.map +1 -1
- package/dist/index.barrel.js +2 -4
- package/dist/index.d.ts +143 -25
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1883 -310
- package/dist/persona/PersonaElement.d.ts +10 -0
- package/dist/persona/PersonaElement.d.ts.map +1 -1
- package/dist/persona/PersonaElement.js +55 -32
- package/dist/persona/PersonaElementManager.d.ts.map +1 -1
- package/dist/persona/PersonaElementManager.js +13 -11
- package/dist/persona/PersonaLoader.d.ts.map +1 -1
- package/dist/persona/PersonaLoader.js +8 -2
- package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
- package/dist/persona/export-import/PersonaImporter.js +24 -5
- package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
- package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
- package/dist/persona/export-import/PersonaSharer.js +198 -22
- package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
- package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
- package/dist/portfolio/DefaultElementProvider.js +499 -7
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
- package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
- package/dist/portfolio/MigrationManager.d.ts.map +1 -1
- package/dist/portfolio/MigrationManager.js +136 -3
- package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
- package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioIndexManager.js +478 -0
- package/dist/portfolio/PortfolioManager.d.ts +5 -0
- package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioManager.js +61 -20
- package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
- package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioRepoManager.js +337 -0
- package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
- package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
- package/dist/portfolio/UnifiedIndexManager.js +1434 -0
- package/dist/portfolio/index.d.ts +15 -0
- package/dist/portfolio/index.d.ts.map +1 -0
- package/dist/portfolio/index.js +15 -0
- package/dist/portfolio/types.d.ts +7 -0
- package/dist/portfolio/types.d.ts.map +1 -1
- package/dist/portfolio/types.js +6 -1
- package/dist/security/InputValidator.d.ts.map +1 -1
- package/dist/security/InputValidator.js +50 -48
- package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
- package/dist/security/audit/SecurityAuditor.js +17 -9
- package/dist/security/audit/config/suppressions.d.ts.map +1 -1
- package/dist/security/audit/config/suppressions.js +19 -3
- package/dist/security/contentValidator.d.ts +2 -0
- package/dist/security/contentValidator.d.ts.map +1 -1
- package/dist/security/contentValidator.js +115 -4
- package/dist/security/secureYamlParser.d.ts +1 -0
- package/dist/security/secureYamlParser.d.ts.map +1 -1
- package/dist/security/secureYamlParser.js +29 -7
- package/dist/security/securityMonitor.d.ts +1 -1
- package/dist/security/securityMonitor.d.ts.map +1 -1
- package/dist/security/securityMonitor.js +1 -1
- package/dist/security/tokenManager.d.ts +1 -1
- package/dist/security/tokenManager.d.ts.map +1 -1
- package/dist/security/tokenManager.js +30 -10
- package/dist/server/ServerSetup.d.ts +22 -2
- package/dist/server/ServerSetup.d.ts.map +1 -1
- package/dist/server/ServerSetup.js +77 -12
- package/dist/server/tools/AuthTools.d.ts.map +1 -1
- package/dist/server/tools/AuthTools.js +33 -1
- package/dist/server/tools/BuildInfoTools.d.ts +25 -0
- package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
- package/dist/server/tools/BuildInfoTools.js +36 -0
- package/dist/server/tools/CollectionTools.d.ts.map +1 -1
- package/dist/server/tools/CollectionTools.js +55 -46
- package/dist/server/tools/ConfigTools.d.ts.map +1 -1
- package/dist/server/tools/ConfigTools.js +29 -1
- package/dist/server/tools/PersonaTools.d.ts +4 -2
- package/dist/server/tools/PersonaTools.d.ts.map +1 -1
- package/dist/server/tools/PersonaTools.js +5 -152
- package/dist/server/tools/PortfolioTools.d.ts +12 -0
- package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
- package/dist/server/tools/PortfolioTools.js +221 -0
- package/dist/server/tools/index.d.ts +3 -1
- package/dist/server/tools/index.d.ts.map +1 -1
- package/dist/server/tools/index.js +4 -2
- package/dist/server/types.d.ts +40 -5
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/types.js +1 -1
- package/dist/services/BuildInfoService.d.ts +84 -0
- package/dist/services/BuildInfoService.d.ts.map +1 -0
- package/dist/services/BuildInfoService.js +271 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
- package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
- package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
- package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
- package/dist/tools/portfolio/types.d.ts +41 -0
- package/dist/tools/portfolio/types.d.ts.map +1 -0
- package/dist/tools/portfolio/types.js +15 -0
- package/dist/types/collection.d.ts +51 -0
- package/dist/types/collection.d.ts.map +1 -1
- package/dist/types/collection.js +1 -1
- package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
- package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
- package/dist/utils/EarlyTerminationSearch.js +164 -0
- package/dist/utils/ErrorHandler.d.ts +86 -0
- package/dist/utils/ErrorHandler.d.ts.map +1 -0
- package/dist/utils/ErrorHandler.js +201 -0
- package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
- package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
- package/dist/utils/FileDiscoveryUtil.js +169 -0
- package/dist/utils/GitHubRateLimiter.d.ts +88 -0
- package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
- package/dist/utils/GitHubRateLimiter.js +315 -0
- package/dist/utils/PerformanceMonitor.d.ts +134 -0
- package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
- package/dist/utils/PerformanceMonitor.js +347 -0
- package/dist/utils/RateLimiter.d.ts.map +1 -0
- package/dist/utils/RateLimiter.js +172 -0
- package/dist/utils/SecureDownloader.d.ts +241 -0
- package/dist/utils/SecureDownloader.d.ts.map +1 -0
- package/dist/utils/SecureDownloader.js +759 -0
- package/dist/utils/ToolCache.d.ts +82 -0
- package/dist/utils/ToolCache.d.ts.map +1 -0
- package/dist/utils/ToolCache.js +196 -0
- package/dist/utils/errorCodes.d.ts +136 -0
- package/dist/utils/errorCodes.d.ts.map +1 -0
- package/dist/utils/errorCodes.js +87 -0
- package/dist/utils/index.d.ts +3 -0
- package/dist/utils/index.d.ts.map +1 -1
- package/dist/utils/index.js +4 -1
- package/dist/utils/installation.d.ts +1 -1
- package/dist/utils/installation.d.ts.map +1 -1
- package/dist/utils/installation.js +9 -8
- package/dist/utils/searchUtils.d.ts +31 -0
- package/dist/utils/searchUtils.d.ts.map +1 -1
- package/dist/utils/searchUtils.js +62 -1
- package/package.json +17 -7
- package/dist/config/updateConfig.d.ts +0 -84
- package/dist/config/updateConfig.d.ts.map +0 -1
- package/dist/config/updateConfig.js +0 -148
- package/dist/server/tools/UpdateTools.d.ts +0 -10
- package/dist/server/tools/UpdateTools.d.ts.map +0 -1
- package/dist/server/tools/UpdateTools.js +0 -85
- package/dist/update/BackupManager.d.ts +0 -63
- package/dist/update/BackupManager.d.ts.map +0 -1
- package/dist/update/BackupManager.js +0 -370
- package/dist/update/DependencyChecker.d.ts +0 -41
- package/dist/update/DependencyChecker.d.ts.map +0 -1
- package/dist/update/DependencyChecker.js +0 -132
- package/dist/update/RateLimiter.d.ts.map +0 -1
- package/dist/update/RateLimiter.js +0 -172
- package/dist/update/SignatureVerifier.d.ts +0 -71
- package/dist/update/SignatureVerifier.d.ts.map +0 -1
- package/dist/update/SignatureVerifier.js +0 -214
- package/dist/update/UpdateChecker.d.ts +0 -132
- package/dist/update/UpdateChecker.d.ts.map +0 -1
- package/dist/update/UpdateChecker.js +0 -506
- package/dist/update/UpdateManager.d.ts +0 -60
- package/dist/update/UpdateManager.d.ts.map +0 -1
- package/dist/update/UpdateManager.js +0 -730
- package/dist/update/VersionManager.d.ts +0 -31
- package/dist/update/VersionManager.d.ts.map +0 -1
- package/dist/update/VersionManager.js +0 -181
- package/dist/update/index.d.ts +0 -9
- package/dist/update/index.d.ts.map +0 -1
- package/dist/update/index.js +0 -9
- /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ConfigManager - Thread-safe singleton for persistent configuration
|
|
3
|
+
*
|
|
4
|
+
* Handles OAuth client ID storage for Claude Desktop integration.
|
|
5
|
+
* Stores config in ~/.dollhouse/config.json with proper permissions.
|
|
6
|
+
* Prefers environment variables over config file values.
|
|
7
|
+
*/
|
|
8
|
+
interface ConfigData {
|
|
9
|
+
version: string;
|
|
10
|
+
oauth?: {
|
|
11
|
+
githubClientId?: string;
|
|
12
|
+
};
|
|
13
|
+
[key: string]: any;
|
|
14
|
+
}
|
|
15
|
+
export declare class ConfigManager {
|
|
16
|
+
private static instance;
|
|
17
|
+
private static instanceLock;
|
|
18
|
+
private configDir;
|
|
19
|
+
private configPath;
|
|
20
|
+
private config;
|
|
21
|
+
private constructor();
|
|
22
|
+
/**
|
|
23
|
+
* Thread-safe singleton instance getter
|
|
24
|
+
*/
|
|
25
|
+
static getInstance(): ConfigManager;
|
|
26
|
+
/**
|
|
27
|
+
* Attempt to repair file permissions if they're incorrect
|
|
28
|
+
* This helps with error recovery in permission-related issues
|
|
29
|
+
*/
|
|
30
|
+
private repairPermissions;
|
|
31
|
+
/**
|
|
32
|
+
* Load configuration from file system
|
|
33
|
+
*/
|
|
34
|
+
loadConfig(): Promise<void>;
|
|
35
|
+
/**
|
|
36
|
+
* Get GitHub OAuth client ID
|
|
37
|
+
* Environment variable takes precedence over config file
|
|
38
|
+
*/
|
|
39
|
+
getGitHubClientId(): string | null;
|
|
40
|
+
/**
|
|
41
|
+
* Set GitHub OAuth client ID in config file
|
|
42
|
+
*/
|
|
43
|
+
setGitHubClientId(clientId: string): Promise<void>;
|
|
44
|
+
/**
|
|
45
|
+
* Get a copy of the current configuration
|
|
46
|
+
* @returns A defensive copy of the configuration object
|
|
47
|
+
*/
|
|
48
|
+
getConfig(): ConfigData;
|
|
49
|
+
/**
|
|
50
|
+
* Update the entire configuration
|
|
51
|
+
* @param newConfig The new configuration to set
|
|
52
|
+
*/
|
|
53
|
+
updateConfig(newConfig: ConfigData): Promise<void>;
|
|
54
|
+
/**
|
|
55
|
+
* Validate GitHub OAuth client ID format
|
|
56
|
+
* Client IDs start with "Ov23li" followed by at least 14 alphanumeric characters
|
|
57
|
+
*
|
|
58
|
+
* @param clientId - The client ID to validate
|
|
59
|
+
* @returns true if valid, false otherwise
|
|
60
|
+
*
|
|
61
|
+
* @example
|
|
62
|
+
* ConfigManager.validateClientId("Ov23liABCDEFGHIJKLMN123456") // true
|
|
63
|
+
* ConfigManager.validateClientId("invalid") // false
|
|
64
|
+
* ConfigManager.validateClientId("Ov23li") // false (too short)
|
|
65
|
+
* ConfigManager.validateClientId("Xv23liABCDEFGHIJKLMN") // false (wrong prefix)
|
|
66
|
+
*/
|
|
67
|
+
static validateClientId(clientId: any): boolean;
|
|
68
|
+
/**
|
|
69
|
+
* Ensure config directory exists with proper permissions
|
|
70
|
+
*/
|
|
71
|
+
private ensureConfigDirectory;
|
|
72
|
+
/**
|
|
73
|
+
* Save config using atomic file writes
|
|
74
|
+
*/
|
|
75
|
+
private saveConfig;
|
|
76
|
+
}
|
|
77
|
+
export {};
|
|
78
|
+
//# sourceMappingURL=ConfigManager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ConfigManager.d.ts","sourceRoot":"","sources":["../../src/config/ConfigManager.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAMH,UAAU,UAAU;IAClB,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,CAAC,EAAE;QACN,cAAc,CAAC,EAAE,MAAM,CAAC;KACzB,CAAC;IACF,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAC;CACpB;AAED,qBAAa,aAAa;IACxB,OAAO,CAAC,MAAM,CAAC,QAAQ,CAA8B;IACrD,OAAO,CAAC,MAAM,CAAC,YAAY,CAAkB;IAE7C,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,MAAM,CAAa;IAE3B,OAAO;IAWP;;OAEG;WACW,WAAW,IAAI,aAAa;IAyB1C;;;OAGG;YACW,iBAAiB;IAkB/B;;OAEG;IACU,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAuCxC;;;OAGG;IACI,iBAAiB,IAAI,MAAM,GAAG,IAAI;IAWzC;;OAEG;IACU,iBAAiB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAgB/D;;;OAGG;IACI,SAAS,IAAI,UAAU;IAI9B;;;OAGG;IACU,YAAY,CAAC,SAAS,EAAE,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC;IAK/D;;;;;;;;;;;;OAYG;WACW,gBAAgB,CAAC,QAAQ,EAAE,GAAG,GAAG,OAAO;IAUtD;;OAEG;YACW,qBAAqB;IAWnC;;OAEG;YACW,UAAU;CAuBzB"}
|
|
@@ -0,0 +1,216 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ConfigManager - Thread-safe singleton for persistent configuration
|
|
3
|
+
*
|
|
4
|
+
* Handles OAuth client ID storage for Claude Desktop integration.
|
|
5
|
+
* Stores config in ~/.dollhouse/config.json with proper permissions.
|
|
6
|
+
* Prefers environment variables over config file values.
|
|
7
|
+
*/
|
|
8
|
+
import * as fs from 'fs/promises';
|
|
9
|
+
import * as path from 'path';
|
|
10
|
+
import * as os from 'os';
|
|
11
|
+
export class ConfigManager {
|
|
12
|
+
static instance = null;
|
|
13
|
+
static instanceLock = false;
|
|
14
|
+
configDir;
|
|
15
|
+
configPath;
|
|
16
|
+
config;
|
|
17
|
+
constructor() {
|
|
18
|
+
// Initialize paths
|
|
19
|
+
this.configDir = path.join(os.homedir(), '.dollhouse');
|
|
20
|
+
this.configPath = path.join(this.configDir, 'config.json');
|
|
21
|
+
// Initialize with default config
|
|
22
|
+
this.config = {
|
|
23
|
+
version: '1.0.0'
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Thread-safe singleton instance getter
|
|
28
|
+
*/
|
|
29
|
+
static getInstance() {
|
|
30
|
+
if (ConfigManager.instance) {
|
|
31
|
+
return ConfigManager.instance;
|
|
32
|
+
}
|
|
33
|
+
// Simple locking mechanism to prevent race conditions
|
|
34
|
+
if (ConfigManager.instanceLock) {
|
|
35
|
+
// Wait for lock to be released, then return the instance
|
|
36
|
+
while (ConfigManager.instanceLock && !ConfigManager.instance) {
|
|
37
|
+
// In a real scenario with async operations, this would be more sophisticated
|
|
38
|
+
// But for the test cases, this simple approach works
|
|
39
|
+
}
|
|
40
|
+
return ConfigManager.instance;
|
|
41
|
+
}
|
|
42
|
+
ConfigManager.instanceLock = true;
|
|
43
|
+
if (!ConfigManager.instance) {
|
|
44
|
+
ConfigManager.instance = new ConfigManager();
|
|
45
|
+
}
|
|
46
|
+
ConfigManager.instanceLock = false;
|
|
47
|
+
return ConfigManager.instance;
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Attempt to repair file permissions if they're incorrect
|
|
51
|
+
* This helps with error recovery in permission-related issues
|
|
52
|
+
*/
|
|
53
|
+
async repairPermissions() {
|
|
54
|
+
try {
|
|
55
|
+
// Try to fix directory permissions
|
|
56
|
+
await fs.chmod(this.configDir, 0o700);
|
|
57
|
+
// Try to fix file permissions if it exists
|
|
58
|
+
try {
|
|
59
|
+
await fs.access(this.configPath);
|
|
60
|
+
await fs.chmod(this.configPath, 0o600);
|
|
61
|
+
}
|
|
62
|
+
catch {
|
|
63
|
+
// File doesn't exist, that's OK
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
catch (error) {
|
|
67
|
+
// Log but don't fail - this is best-effort recovery
|
|
68
|
+
// We don't have a logger here, so we'll silently continue
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Load configuration from file system
|
|
73
|
+
*/
|
|
74
|
+
async loadConfig() {
|
|
75
|
+
try {
|
|
76
|
+
// Try to read existing config file
|
|
77
|
+
const configContent = await fs.readFile(this.configPath, 'utf-8');
|
|
78
|
+
try {
|
|
79
|
+
this.config = JSON.parse(configContent);
|
|
80
|
+
}
|
|
81
|
+
catch (parseError) {
|
|
82
|
+
// Handle corrupted JSON - create new config
|
|
83
|
+
console.warn('Config file corrupted, creating new config');
|
|
84
|
+
this.config = { version: '1.0.0' };
|
|
85
|
+
await this.saveConfig();
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
catch (error) {
|
|
89
|
+
if (error.code === 'ENOENT') {
|
|
90
|
+
// Config file doesn't exist, create directory and file
|
|
91
|
+
await this.ensureConfigDirectory();
|
|
92
|
+
await this.saveConfig();
|
|
93
|
+
}
|
|
94
|
+
else if (error.code === 'EACCES' || error.code === 'EPERM') {
|
|
95
|
+
// Permission denied - attempt repair
|
|
96
|
+
await this.repairPermissions();
|
|
97
|
+
// Try once more after repair attempt
|
|
98
|
+
try {
|
|
99
|
+
const configContent = await fs.readFile(this.configPath, 'utf-8');
|
|
100
|
+
this.config = JSON.parse(configContent);
|
|
101
|
+
}
|
|
102
|
+
catch (retryError) {
|
|
103
|
+
// Still failing, throw original error with helpful message
|
|
104
|
+
throw new Error(`Permission denied accessing config at ${this.configPath}. ` +
|
|
105
|
+
`Please check file permissions or run with appropriate privileges.`);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
else {
|
|
109
|
+
throw error;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
/**
|
|
114
|
+
* Get GitHub OAuth client ID
|
|
115
|
+
* Environment variable takes precedence over config file
|
|
116
|
+
*/
|
|
117
|
+
getGitHubClientId() {
|
|
118
|
+
// Check environment variable first
|
|
119
|
+
const envClientId = process.env.DOLLHOUSE_GITHUB_CLIENT_ID;
|
|
120
|
+
if (envClientId) {
|
|
121
|
+
return envClientId;
|
|
122
|
+
}
|
|
123
|
+
// Fall back to config file
|
|
124
|
+
return this.config.oauth?.githubClientId || null;
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Set GitHub OAuth client ID in config file
|
|
128
|
+
*/
|
|
129
|
+
async setGitHubClientId(clientId) {
|
|
130
|
+
if (!ConfigManager.validateClientId(clientId)) {
|
|
131
|
+
throw new Error(`Invalid GitHub client ID format. Expected format: Ov23li followed by at least 14 alphanumeric characters (e.g., Ov23liABCDEFGHIJKLMN)`);
|
|
132
|
+
}
|
|
133
|
+
// Ensure oauth object exists
|
|
134
|
+
if (!this.config.oauth) {
|
|
135
|
+
this.config.oauth = {};
|
|
136
|
+
}
|
|
137
|
+
this.config.oauth.githubClientId = clientId;
|
|
138
|
+
await this.saveConfig();
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Get a copy of the current configuration
|
|
142
|
+
* @returns A defensive copy of the configuration object
|
|
143
|
+
*/
|
|
144
|
+
getConfig() {
|
|
145
|
+
return { ...this.config };
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* Update the entire configuration
|
|
149
|
+
* @param newConfig The new configuration to set
|
|
150
|
+
*/
|
|
151
|
+
async updateConfig(newConfig) {
|
|
152
|
+
this.config = { ...newConfig };
|
|
153
|
+
await this.saveConfig();
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Validate GitHub OAuth client ID format
|
|
157
|
+
* Client IDs start with "Ov23li" followed by at least 14 alphanumeric characters
|
|
158
|
+
*
|
|
159
|
+
* @param clientId - The client ID to validate
|
|
160
|
+
* @returns true if valid, false otherwise
|
|
161
|
+
*
|
|
162
|
+
* @example
|
|
163
|
+
* ConfigManager.validateClientId("Ov23liABCDEFGHIJKLMN123456") // true
|
|
164
|
+
* ConfigManager.validateClientId("invalid") // false
|
|
165
|
+
* ConfigManager.validateClientId("Ov23li") // false (too short)
|
|
166
|
+
* ConfigManager.validateClientId("Xv23liABCDEFGHIJKLMN") // false (wrong prefix)
|
|
167
|
+
*/
|
|
168
|
+
static validateClientId(clientId) {
|
|
169
|
+
if (typeof clientId !== 'string' || !clientId) {
|
|
170
|
+
return false;
|
|
171
|
+
}
|
|
172
|
+
// GitHub OAuth client IDs follow the pattern: Ov23li[A-Za-z0-9]{14,}
|
|
173
|
+
const clientIdPattern = /^Ov23li[A-Za-z0-9]{14,}$/;
|
|
174
|
+
return clientIdPattern.test(clientId);
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* Ensure config directory exists with proper permissions
|
|
178
|
+
*/
|
|
179
|
+
async ensureConfigDirectory() {
|
|
180
|
+
try {
|
|
181
|
+
await fs.mkdir(this.configDir, { recursive: true, mode: 0o700 });
|
|
182
|
+
}
|
|
183
|
+
catch (error) {
|
|
184
|
+
if (error.code === 'EACCES') {
|
|
185
|
+
throw new Error(`Permission denied creating config directory: ${this.configDir}`);
|
|
186
|
+
}
|
|
187
|
+
throw error;
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
/**
|
|
191
|
+
* Save config using atomic file writes
|
|
192
|
+
*/
|
|
193
|
+
async saveConfig() {
|
|
194
|
+
await this.ensureConfigDirectory();
|
|
195
|
+
// Use atomic write: write to temp file, then rename
|
|
196
|
+
const tempPath = this.configPath + '.tmp';
|
|
197
|
+
const configContent = JSON.stringify(this.config, null, 2);
|
|
198
|
+
try {
|
|
199
|
+
// Write to temp file first
|
|
200
|
+
await fs.writeFile(tempPath, configContent, { mode: 0o600 });
|
|
201
|
+
// Atomic rename
|
|
202
|
+
await fs.rename(tempPath, this.configPath);
|
|
203
|
+
}
|
|
204
|
+
catch (error) {
|
|
205
|
+
// Clean up temp file if it exists
|
|
206
|
+
try {
|
|
207
|
+
await fs.unlink(tempPath);
|
|
208
|
+
}
|
|
209
|
+
catch {
|
|
210
|
+
// Ignore cleanup errors
|
|
211
|
+
}
|
|
212
|
+
throw error;
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ConfigManager.js","sourceRoot":"","sources":["../../src/config/ConfigManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ConfigManager - Thread-safe singleton for persistent configuration\n * \n * Handles OAuth client ID storage for Claude Desktop integration.\n * Stores config in ~/.dollhouse/config.json with proper permissions.\n * Prefers environment variables over config file values.\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport * as os from 'os';\n\ninterface ConfigData {\n  version: string;\n  oauth?: {\n    githubClientId?: string;\n  };\n  [key: string]: any; // Allow unknown fields to be preserved\n}\n\nexport class ConfigManager {\n  private static instance: ConfigManager | null = null;\n  private static instanceLock: boolean = false;\n\n  private configDir: string;\n  private configPath: string;\n  private config: ConfigData;\n\n  private constructor() {\n    // Initialize paths\n    this.configDir = path.join(os.homedir(), '.dollhouse');\n    this.configPath = path.join(this.configDir, 'config.json');\n    \n    // Initialize with default config\n    this.config = {\n      version: '1.0.0'\n    };\n  }\n\n  /**\n   * Thread-safe singleton instance getter\n   */\n  public static getInstance(): ConfigManager {\n    if (ConfigManager.instance) {\n      return ConfigManager.instance;\n    }\n\n    // Simple locking mechanism to prevent race conditions\n    if (ConfigManager.instanceLock) {\n      // Wait for lock to be released, then return the instance\n      while (ConfigManager.instanceLock && !ConfigManager.instance) {\n        // In a real scenario with async operations, this would be more sophisticated\n        // But for the test cases, this simple approach works\n      }\n      return ConfigManager.instance!;\n    }\n\n    ConfigManager.instanceLock = true;\n    \n    if (!ConfigManager.instance) {\n      ConfigManager.instance = new ConfigManager();\n    }\n    \n    ConfigManager.instanceLock = false;\n    return ConfigManager.instance;\n  }\n\n  /**\n   * Attempt to repair file permissions if they're incorrect\n   * This helps with error recovery in permission-related issues\n   */\n  private async repairPermissions(): Promise<void> {\n    try {\n      // Try to fix directory permissions\n      await fs.chmod(this.configDir, 0o700);\n      \n      // Try to fix file permissions if it exists\n      try {\n        await fs.access(this.configPath);\n        await fs.chmod(this.configPath, 0o600);\n      } catch {\n        // File doesn't exist, that's OK\n      }\n    } catch (error) {\n      // Log but don't fail - this is best-effort recovery\n      // We don't have a logger here, so we'll silently continue\n    }\n  }\n\n  /**\n   * Load configuration from file system\n   */\n  public async loadConfig(): Promise<void> {\n    try {\n      // Try to read existing config file\n      const configContent = await fs.readFile(this.configPath, 'utf-8');\n      \n      try {\n        this.config = JSON.parse(configContent);\n      } catch (parseError) {\n        // Handle corrupted JSON - create new config\n        console.warn('Config file corrupted, creating new config');\n        this.config = { version: '1.0.0' };\n        await this.saveConfig();\n      }\n    } catch (error: any) {\n      if (error.code === 'ENOENT') {\n        // Config file doesn't exist, create directory and file\n        await this.ensureConfigDirectory();\n        await this.saveConfig();\n      } else if (error.code === 'EACCES' || error.code === 'EPERM') {\n        // Permission denied - attempt repair\n        await this.repairPermissions();\n        \n        // Try once more after repair attempt\n        try {\n          const configContent = await fs.readFile(this.configPath, 'utf-8');\n          this.config = JSON.parse(configContent);\n        } catch (retryError: any) {\n          // Still failing, throw original error with helpful message\n          throw new Error(\n            `Permission denied accessing config at ${this.configPath}. ` +\n            `Please check file permissions or run with appropriate privileges.`\n          );\n        }\n      } else {\n        throw error;\n      }\n    }\n  }\n\n  /**\n   * Get GitHub OAuth client ID\n   * Environment variable takes precedence over config file\n   */\n  public getGitHubClientId(): string | null {\n    // Check environment variable first\n    const envClientId = process.env.DOLLHOUSE_GITHUB_CLIENT_ID;\n    if (envClientId) {\n      return envClientId;\n    }\n\n    // Fall back to config file\n    return this.config.oauth?.githubClientId || null;\n  }\n\n  /**\n   * Set GitHub OAuth client ID in config file\n   */\n  public async setGitHubClientId(clientId: string): Promise<void> {\n    if (!ConfigManager.validateClientId(clientId)) {\n      throw new Error(\n        `Invalid GitHub client ID format. Expected format: Ov23li followed by at least 14 alphanumeric characters (e.g., Ov23liABCDEFGHIJKLMN)`\n      );\n    }\n\n    // Ensure oauth object exists\n    if (!this.config.oauth) {\n      this.config.oauth = {};\n    }\n\n    this.config.oauth.githubClientId = clientId;\n    await this.saveConfig();\n  }\n\n  /**\n   * Get a copy of the current configuration\n   * @returns A defensive copy of the configuration object\n   */\n  public getConfig(): ConfigData {\n    return { ...this.config };\n  }\n\n  /**\n   * Update the entire configuration\n   * @param newConfig The new configuration to set\n   */\n  public async updateConfig(newConfig: ConfigData): Promise<void> {\n    this.config = { ...newConfig };\n    await this.saveConfig();\n  }\n\n  /**\n   * Validate GitHub OAuth client ID format\n   * Client IDs start with \"Ov23li\" followed by at least 14 alphanumeric characters\n   * \n   * @param clientId - The client ID to validate\n   * @returns true if valid, false otherwise\n   * \n   * @example\n   * ConfigManager.validateClientId(\"Ov23liABCDEFGHIJKLMN123456\") // true\n   * ConfigManager.validateClientId(\"invalid\") // false\n   * ConfigManager.validateClientId(\"Ov23li\") // false (too short)\n   * ConfigManager.validateClientId(\"Xv23liABCDEFGHIJKLMN\") // false (wrong prefix)\n   */\n  public static validateClientId(clientId: any): boolean {\n    if (typeof clientId !== 'string' || !clientId) {\n      return false;\n    }\n\n    // GitHub OAuth client IDs follow the pattern: Ov23li[A-Za-z0-9]{14,}\n    const clientIdPattern = /^Ov23li[A-Za-z0-9]{14,}$/;\n    return clientIdPattern.test(clientId);\n  }\n\n  /**\n   * Ensure config directory exists with proper permissions\n   */\n  private async ensureConfigDirectory(): Promise<void> {\n    try {\n      await fs.mkdir(this.configDir, { recursive: true, mode: 0o700 });\n    } catch (error: any) {\n      if (error.code === 'EACCES') {\n        throw new Error(`Permission denied creating config directory: ${this.configDir}`);\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Save config using atomic file writes\n   */\n  private async saveConfig(): Promise<void> {\n    await this.ensureConfigDirectory();\n    \n    // Use atomic write: write to temp file, then rename\n    const tempPath = this.configPath + '.tmp';\n    const configContent = JSON.stringify(this.config, null, 2);\n    \n    try {\n      // Write to temp file first\n      await fs.writeFile(tempPath, configContent, { mode: 0o600 });\n      \n      // Atomic rename\n      await fs.rename(tempPath, this.configPath);\n    } catch (error) {\n      // Clean up temp file if it exists\n      try {\n        await fs.unlink(tempPath);\n      } catch {\n        // Ignore cleanup errors\n      }\n      throw error;\n    }\n  }\n}"]}
|
|
@@ -0,0 +1,135 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Central configuration for all element types
|
|
3
|
+
*
|
|
4
|
+
* This file serves as the single source of truth for element type configurations.
|
|
5
|
+
* When adding new element types, update this config and use the derived arrays
|
|
6
|
+
* to ensure consistency across the codebase.
|
|
7
|
+
*/
|
|
8
|
+
import { ElementType } from '../portfolio/types.js';
|
|
9
|
+
/**
|
|
10
|
+
* Complete configuration for each element type
|
|
11
|
+
*/
|
|
12
|
+
export declare const ELEMENT_TYPE_CONFIG: {
|
|
13
|
+
readonly personas: {
|
|
14
|
+
readonly plural: "personas";
|
|
15
|
+
readonly directory: "personas";
|
|
16
|
+
readonly mcpSupported: true;
|
|
17
|
+
readonly hasManager: true;
|
|
18
|
+
readonly icon: "👤";
|
|
19
|
+
readonly description: "Behavioral profiles that define AI personality and interaction style";
|
|
20
|
+
};
|
|
21
|
+
readonly skills: {
|
|
22
|
+
readonly plural: "skills";
|
|
23
|
+
readonly directory: "skills";
|
|
24
|
+
readonly mcpSupported: true;
|
|
25
|
+
readonly hasManager: true;
|
|
26
|
+
readonly icon: "🛠️";
|
|
27
|
+
readonly description: "Discrete capabilities for specific tasks";
|
|
28
|
+
};
|
|
29
|
+
readonly templates: {
|
|
30
|
+
readonly plural: "templates";
|
|
31
|
+
readonly directory: "templates";
|
|
32
|
+
readonly mcpSupported: true;
|
|
33
|
+
readonly hasManager: true;
|
|
34
|
+
readonly icon: "📄";
|
|
35
|
+
readonly description: "Reusable content structures with variable substitution";
|
|
36
|
+
};
|
|
37
|
+
readonly agents: {
|
|
38
|
+
readonly plural: "agents";
|
|
39
|
+
readonly directory: "agents";
|
|
40
|
+
readonly mcpSupported: true;
|
|
41
|
+
readonly hasManager: true;
|
|
42
|
+
readonly icon: "🤖";
|
|
43
|
+
readonly description: "Autonomous goal-oriented actors with decision-making capabilities";
|
|
44
|
+
};
|
|
45
|
+
readonly memories: {
|
|
46
|
+
readonly plural: "memories";
|
|
47
|
+
readonly directory: "memories";
|
|
48
|
+
readonly mcpSupported: false;
|
|
49
|
+
readonly hasManager: false;
|
|
50
|
+
readonly icon: "🧠";
|
|
51
|
+
readonly description: "Persistent context storage for continuity and learning";
|
|
52
|
+
};
|
|
53
|
+
readonly ensembles: {
|
|
54
|
+
readonly plural: "ensembles";
|
|
55
|
+
readonly directory: "ensembles";
|
|
56
|
+
readonly mcpSupported: false;
|
|
57
|
+
readonly hasManager: false;
|
|
58
|
+
readonly icon: "🎭";
|
|
59
|
+
readonly description: "Groups of elements working together as a cohesive unit";
|
|
60
|
+
};
|
|
61
|
+
};
|
|
62
|
+
/**
|
|
63
|
+
* Derived arrays for validation and filtering
|
|
64
|
+
* These are automatically generated from the config above
|
|
65
|
+
*/
|
|
66
|
+
export declare const MCP_SUPPORTED_TYPES: ElementType[];
|
|
67
|
+
export declare const VALID_TYPES_ARRAY: ("personas" | "skills" | "templates" | "agents" | "memories" | "ensembles")[];
|
|
68
|
+
export declare const PLURAL_TO_ELEMENT_TYPE_MAP: {
|
|
69
|
+
[k: string]: ElementType;
|
|
70
|
+
};
|
|
71
|
+
export declare const SINGULAR_TO_DIRECTORY_MAP: {
|
|
72
|
+
[k: string]: "personas" | "skills" | "templates" | "agents" | "memories" | "ensembles";
|
|
73
|
+
};
|
|
74
|
+
/**
|
|
75
|
+
* Utility functions
|
|
76
|
+
*/
|
|
77
|
+
export declare function isElementTypeSupported(type: ElementType): boolean;
|
|
78
|
+
export declare function getElementTypeConfig(type: ElementType): {
|
|
79
|
+
readonly plural: "personas";
|
|
80
|
+
readonly directory: "personas";
|
|
81
|
+
readonly mcpSupported: true;
|
|
82
|
+
readonly hasManager: true;
|
|
83
|
+
readonly icon: "👤";
|
|
84
|
+
readonly description: "Behavioral profiles that define AI personality and interaction style";
|
|
85
|
+
} | {
|
|
86
|
+
readonly plural: "skills";
|
|
87
|
+
readonly directory: "skills";
|
|
88
|
+
readonly mcpSupported: true;
|
|
89
|
+
readonly hasManager: true;
|
|
90
|
+
readonly icon: "🛠️";
|
|
91
|
+
readonly description: "Discrete capabilities for specific tasks";
|
|
92
|
+
} | {
|
|
93
|
+
readonly plural: "templates";
|
|
94
|
+
readonly directory: "templates";
|
|
95
|
+
readonly mcpSupported: true;
|
|
96
|
+
readonly hasManager: true;
|
|
97
|
+
readonly icon: "📄";
|
|
98
|
+
readonly description: "Reusable content structures with variable substitution";
|
|
99
|
+
} | {
|
|
100
|
+
readonly plural: "agents";
|
|
101
|
+
readonly directory: "agents";
|
|
102
|
+
readonly mcpSupported: true;
|
|
103
|
+
readonly hasManager: true;
|
|
104
|
+
readonly icon: "🤖";
|
|
105
|
+
readonly description: "Autonomous goal-oriented actors with decision-making capabilities";
|
|
106
|
+
} | {
|
|
107
|
+
readonly plural: "memories";
|
|
108
|
+
readonly directory: "memories";
|
|
109
|
+
readonly mcpSupported: false;
|
|
110
|
+
readonly hasManager: false;
|
|
111
|
+
readonly icon: "🧠";
|
|
112
|
+
readonly description: "Persistent context storage for continuity and learning";
|
|
113
|
+
} | {
|
|
114
|
+
readonly plural: "ensembles";
|
|
115
|
+
readonly directory: "ensembles";
|
|
116
|
+
readonly mcpSupported: false;
|
|
117
|
+
readonly hasManager: false;
|
|
118
|
+
readonly icon: "🎭";
|
|
119
|
+
readonly description: "Groups of elements working together as a cohesive unit";
|
|
120
|
+
};
|
|
121
|
+
export declare function getAllSupportedTypes(): ElementType[];
|
|
122
|
+
export declare function getValidTypesForMCP(): string[];
|
|
123
|
+
/**
|
|
124
|
+
* Migration note: To use this centralized config:
|
|
125
|
+
*
|
|
126
|
+
* 1. Replace hardcoded arrays with imports from this file:
|
|
127
|
+
* - Replace validTypes in src/index.ts with VALID_TYPES_ARRAY
|
|
128
|
+
* - Replace MCP_SUPPORTED_TYPES in CollectionBrowser.ts with import
|
|
129
|
+
* - Replace mapping objects with imports from this file
|
|
130
|
+
*
|
|
131
|
+
* 2. Update components to use utility functions instead of hardcoded checks
|
|
132
|
+
*
|
|
133
|
+
* 3. When adding new types, only update ELEMENT_TYPE_CONFIG above
|
|
134
|
+
*/
|
|
135
|
+
//# sourceMappingURL=element-types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"element-types.d.ts","sourceRoot":"","sources":["../../src/config/element-types.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAEpD;;GAEG;AACH,eAAO,MAAM,mBAAmB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAiDtB,CAAC;AAEX;;;GAGG;AAGH,eAAO,MAAM,mBAAmB,eAEO,CAAC;AAGxC,eAAO,MAAM,iBAAiB,+EAE7B,CAAC;AAGF,eAAO,MAAM,0BAA0B;;CAItC,CAAC;AAGF,eAAO,MAAM,yBAAyB;;CAKrC,CAAC;AAEF;;GAEG;AAEH,wBAAgB,sBAAsB,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAEjE;AAED,wBAAgB,oBAAoB,CAAC,IAAI,EAAE,WAAW;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAErD;AAED,wBAAgB,oBAAoB,IAAI,WAAW,EAAE,CAEpD;AAED,wBAAgB,mBAAmB,IAAI,MAAM,EAAE,CAE9C;AAED;;;;;;;;;;;GAWG"}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Central configuration for all element types
|
|
3
|
+
*
|
|
4
|
+
* This file serves as the single source of truth for element type configurations.
|
|
5
|
+
* When adding new element types, update this config and use the derived arrays
|
|
6
|
+
* to ensure consistency across the codebase.
|
|
7
|
+
*/
|
|
8
|
+
import { ElementType } from '../portfolio/types.js';
|
|
9
|
+
/**
|
|
10
|
+
* Complete configuration for each element type
|
|
11
|
+
*/
|
|
12
|
+
export const ELEMENT_TYPE_CONFIG = {
|
|
13
|
+
[ElementType.PERSONA]: {
|
|
14
|
+
plural: 'personas',
|
|
15
|
+
directory: 'personas',
|
|
16
|
+
mcpSupported: true,
|
|
17
|
+
hasManager: true,
|
|
18
|
+
icon: '👤',
|
|
19
|
+
description: 'Behavioral profiles that define AI personality and interaction style'
|
|
20
|
+
},
|
|
21
|
+
[ElementType.SKILL]: {
|
|
22
|
+
plural: 'skills',
|
|
23
|
+
directory: 'skills',
|
|
24
|
+
mcpSupported: true,
|
|
25
|
+
hasManager: true,
|
|
26
|
+
icon: '🛠️',
|
|
27
|
+
description: 'Discrete capabilities for specific tasks'
|
|
28
|
+
},
|
|
29
|
+
[ElementType.TEMPLATE]: {
|
|
30
|
+
plural: 'templates',
|
|
31
|
+
directory: 'templates',
|
|
32
|
+
mcpSupported: true,
|
|
33
|
+
hasManager: true,
|
|
34
|
+
icon: '📄',
|
|
35
|
+
description: 'Reusable content structures with variable substitution'
|
|
36
|
+
},
|
|
37
|
+
[ElementType.AGENT]: {
|
|
38
|
+
plural: 'agents',
|
|
39
|
+
directory: 'agents',
|
|
40
|
+
mcpSupported: true,
|
|
41
|
+
hasManager: true,
|
|
42
|
+
icon: '🤖',
|
|
43
|
+
description: 'Autonomous goal-oriented actors with decision-making capabilities'
|
|
44
|
+
},
|
|
45
|
+
[ElementType.MEMORY]: {
|
|
46
|
+
plural: 'memories',
|
|
47
|
+
directory: 'memories',
|
|
48
|
+
mcpSupported: false, // Hidden from MCP per Issue #144
|
|
49
|
+
hasManager: false, // Not yet implemented
|
|
50
|
+
icon: '🧠',
|
|
51
|
+
description: 'Persistent context storage for continuity and learning'
|
|
52
|
+
},
|
|
53
|
+
[ElementType.ENSEMBLE]: {
|
|
54
|
+
plural: 'ensembles',
|
|
55
|
+
directory: 'ensembles',
|
|
56
|
+
mcpSupported: false, // Hidden from MCP per Issue #144
|
|
57
|
+
hasManager: false, // Not yet implemented
|
|
58
|
+
icon: '🎭',
|
|
59
|
+
description: 'Groups of elements working together as a cohesive unit'
|
|
60
|
+
}
|
|
61
|
+
};
|
|
62
|
+
/**
|
|
63
|
+
* Derived arrays for validation and filtering
|
|
64
|
+
* These are automatically generated from the config above
|
|
65
|
+
*/
|
|
66
|
+
// Element types that are exposed via MCP tools
|
|
67
|
+
export const MCP_SUPPORTED_TYPES = Object.entries(ELEMENT_TYPE_CONFIG)
|
|
68
|
+
.filter(([, config]) => config.mcpSupported)
|
|
69
|
+
.map(([type]) => type);
|
|
70
|
+
// Plural forms for MCP validation (used in browseCollection validTypes array)
|
|
71
|
+
export const VALID_TYPES_ARRAY = MCP_SUPPORTED_TYPES.map(type => ELEMENT_TYPE_CONFIG[type].plural);
|
|
72
|
+
// Mapping from plural forms to ElementType values
|
|
73
|
+
export const PLURAL_TO_ELEMENT_TYPE_MAP = Object.fromEntries(Object.entries(ELEMENT_TYPE_CONFIG).map(([type, config]) => [
|
|
74
|
+
config.plural, type
|
|
75
|
+
]));
|
|
76
|
+
// Mapping from singular forms to directory names
|
|
77
|
+
export const SINGULAR_TO_DIRECTORY_MAP = Object.fromEntries(Object.entries(ELEMENT_TYPE_CONFIG).map(([type, config]) => [
|
|
78
|
+
type.toLowerCase().replace('s', ''), // Convert 'personas' -> 'persona'
|
|
79
|
+
config.directory
|
|
80
|
+
]));
|
|
81
|
+
/**
|
|
82
|
+
* Utility functions
|
|
83
|
+
*/
|
|
84
|
+
export function isElementTypeSupported(type) {
|
|
85
|
+
return ELEMENT_TYPE_CONFIG[type]?.mcpSupported ?? false;
|
|
86
|
+
}
|
|
87
|
+
export function getElementTypeConfig(type) {
|
|
88
|
+
return ELEMENT_TYPE_CONFIG[type];
|
|
89
|
+
}
|
|
90
|
+
export function getAllSupportedTypes() {
|
|
91
|
+
return MCP_SUPPORTED_TYPES;
|
|
92
|
+
}
|
|
93
|
+
export function getValidTypesForMCP() {
|
|
94
|
+
return VALID_TYPES_ARRAY;
|
|
95
|
+
}
|
|
96
|
+
/**
|
|
97
|
+
* Migration note: To use this centralized config:
|
|
98
|
+
*
|
|
99
|
+
* 1. Replace hardcoded arrays with imports from this file:
|
|
100
|
+
* - Replace validTypes in src/index.ts with VALID_TYPES_ARRAY
|
|
101
|
+
* - Replace MCP_SUPPORTED_TYPES in CollectionBrowser.ts with import
|
|
102
|
+
* - Replace mapping objects with imports from this file
|
|
103
|
+
*
|
|
104
|
+
* 2. Update components to use utility functions instead of hardcoded checks
|
|
105
|
+
*
|
|
106
|
+
* 3. When adding new types, only update ELEMENT_TYPE_CONFIG above
|
|
107
|
+
*/
|
|
108
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"element-types.js","sourceRoot":"","sources":["../../src/config/element-types.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Central configuration for all element types\n * \n * This file serves as the single source of truth for element type configurations.\n * When adding new element types, update this config and use the derived arrays\n * to ensure consistency across the codebase.\n */\n\nimport { ElementType } from '../portfolio/types.js';\n\n/**\n * Complete configuration for each element type\n */\nexport const ELEMENT_TYPE_CONFIG = {\n  [ElementType.PERSONA]: {\n    plural: 'personas',\n    directory: 'personas',\n    mcpSupported: true,\n    hasManager: true,\n    icon: '👤',\n    description: 'Behavioral profiles that define AI personality and interaction style'\n  },\n  [ElementType.SKILL]: {\n    plural: 'skills',\n    directory: 'skills', \n    mcpSupported: true,\n    hasManager: true,\n    icon: '🛠️',\n    description: 'Discrete capabilities for specific tasks'\n  },\n  [ElementType.TEMPLATE]: {\n    plural: 'templates',\n    directory: 'templates',\n    mcpSupported: true,\n    hasManager: true,\n    icon: '📄',\n    description: 'Reusable content structures with variable substitution'\n  },\n  [ElementType.AGENT]: {\n    plural: 'agents',\n    directory: 'agents',\n    mcpSupported: true,\n    hasManager: true,\n    icon: '🤖',\n    description: 'Autonomous goal-oriented actors with decision-making capabilities'\n  },\n  [ElementType.MEMORY]: {\n    plural: 'memories',\n    directory: 'memories',\n    mcpSupported: false, // Hidden from MCP per Issue #144\n    hasManager: false,   // Not yet implemented\n    icon: '🧠',\n    description: 'Persistent context storage for continuity and learning'\n  },\n  [ElementType.ENSEMBLE]: {\n    plural: 'ensembles',\n    directory: 'ensembles',\n    mcpSupported: false, // Hidden from MCP per Issue #144\n    hasManager: false,   // Not yet implemented\n    icon: '🎭',\n    description: 'Groups of elements working together as a cohesive unit'\n  }\n} as const;\n\n/**\n * Derived arrays for validation and filtering\n * These are automatically generated from the config above\n */\n\n// Element types that are exposed via MCP tools\nexport const MCP_SUPPORTED_TYPES = Object.entries(ELEMENT_TYPE_CONFIG)\n  .filter(([, config]) => config.mcpSupported)\n  .map(([type]) => type as ElementType);\n\n// Plural forms for MCP validation (used in browseCollection validTypes array)\nexport const VALID_TYPES_ARRAY = MCP_SUPPORTED_TYPES.map(\n  type => ELEMENT_TYPE_CONFIG[type].plural\n);\n\n// Mapping from plural forms to ElementType values\nexport const PLURAL_TO_ELEMENT_TYPE_MAP = Object.fromEntries(\n  Object.entries(ELEMENT_TYPE_CONFIG).map(([type, config]) => [\n    config.plural, type as ElementType\n  ])\n);\n\n// Mapping from singular forms to directory names  \nexport const SINGULAR_TO_DIRECTORY_MAP = Object.fromEntries(\n  Object.entries(ELEMENT_TYPE_CONFIG).map(([type, config]) => [\n    type.toLowerCase().replace('s', ''), // Convert 'personas' -> 'persona' \n    config.directory\n  ])\n);\n\n/**\n * Utility functions\n */\n\nexport function isElementTypeSupported(type: ElementType): boolean {\n  return ELEMENT_TYPE_CONFIG[type]?.mcpSupported ?? false;\n}\n\nexport function getElementTypeConfig(type: ElementType) {\n  return ELEMENT_TYPE_CONFIG[type];\n}\n\nexport function getAllSupportedTypes(): ElementType[] {\n  return MCP_SUPPORTED_TYPES;\n}\n\nexport function getValidTypesForMCP(): string[] {\n  return VALID_TYPES_ARRAY;\n}\n\n/**\n * Migration note: To use this centralized config:\n * \n * 1. Replace hardcoded arrays with imports from this file:\n *    - Replace validTypes in src/index.ts with VALID_TYPES_ARRAY\n *    - Replace MCP_SUPPORTED_TYPES in CollectionBrowser.ts with import\n *    - Replace mapping objects with imports from this file\n * \n * 2. Update components to use utility functions instead of hardcoded checks\n * \n * 3. When adding new types, only update ELEMENT_TYPE_CONFIG above\n */"]}
|
package/dist/config/index.d.ts
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/config/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,cAAc,gBAAgB,CAAC;AAC/B,cAAc,uBAAuB,CAAC"}
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/config/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,cAAc,gBAAgB,CAAC;AAC/B,cAAc,uBAAuB,CAAC;AACtC,cAAc,oBAAoB,CAAC;AACnC,cAAc,0BAA0B,CAAC"}
|
package/dist/config/index.js
CHANGED
|
@@ -3,4 +3,6 @@
|
|
|
3
3
|
*/
|
|
4
4
|
export * from './constants.js';
|
|
5
5
|
export * from './indicator-config.js';
|
|
6
|
-
|
|
6
|
+
export * from './element-types.js';
|
|
7
|
+
export * from './portfolio-constants.js';
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29uZmlnL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsb0JBQW9CLENBQUM7QUFDbkMsY0FBYywwQkFBMEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ2VudHJhbCBleHBvcnQgcG9pbnQgZm9yIGNvbmZpZ3VyYXRpb25cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2NvbnN0YW50cy5qcyc7XG5leHBvcnQgKiBmcm9tICcuL2luZGljYXRvci1jb25maWcuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9lbGVtZW50LXR5cGVzLmpzJztcbmV4cG9ydCAqIGZyb20gJy4vcG9ydGZvbGlvLWNvbnN0YW50cy5qcyc7Il19
|