@dollhousemcp/mcp-server 1.5.2 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/CHANGELOG.md +56 -0
  2. package/README.md +494 -111
  3. package/data/agents/code-reviewer.md +8 -1
  4. package/data/agents/research-assistant.md +8 -1
  5. package/data/agents/task-manager.md +8 -1
  6. package/data/ensembles/business-advisor.md +8 -1
  7. package/data/ensembles/creative-studio.md +8 -1
  8. package/data/ensembles/development-team.md +8 -1
  9. package/data/ensembles/security-analysis-team.md +8 -1
  10. package/data/memories/conversation-history.md +8 -1
  11. package/data/memories/learning-progress.md +8 -1
  12. package/data/memories/project-context.md +8 -1
  13. package/data/personas/business-consultant.md +8 -1
  14. package/data/personas/creative-writer.md +8 -1
  15. package/data/personas/debug-detective.md +8 -1
  16. package/data/personas/eli5-explainer.md +8 -1
  17. package/data/personas/security-analyst.md +8 -1
  18. package/data/personas/technical-analyst.md +8 -1
  19. package/data/skills/code-review.md +8 -1
  20. package/data/skills/creative-writing.md +8 -1
  21. package/data/skills/data-analysis.md +8 -1
  22. package/data/skills/penetration-testing.md +8 -1
  23. package/data/skills/research.md +8 -1
  24. package/data/skills/threat-modeling.md +8 -1
  25. package/data/skills/translation.md +8 -1
  26. package/data/templates/code-documentation.md +8 -1
  27. package/data/templates/email-professional.md +8 -1
  28. package/data/templates/meeting-notes.md +8 -1
  29. package/data/templates/penetration-test-report.md +8 -1
  30. package/data/templates/project-brief.md +8 -1
  31. package/data/templates/report-executive.md +8 -1
  32. package/data/templates/security-vulnerability-report.md +8 -1
  33. package/data/templates/threat-assessment-report.md +8 -1
  34. package/dist/auth/GitHubAuthManager.d.ts +6 -1
  35. package/dist/auth/GitHubAuthManager.d.ts.map +1 -1
  36. package/dist/auth/GitHubAuthManager.js +45 -18
  37. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts +98 -0
  38. package/dist/benchmarks/IndexPerformanceBenchmark.d.ts.map +1 -0
  39. package/dist/benchmarks/IndexPerformanceBenchmark.js +531 -0
  40. package/dist/cache/CollectionCache.d.ts.map +1 -1
  41. package/dist/cache/CollectionCache.js +13 -3
  42. package/dist/cache/CollectionIndexCache.d.ts +77 -0
  43. package/dist/cache/CollectionIndexCache.d.ts.map +1 -0
  44. package/dist/cache/CollectionIndexCache.js +349 -0
  45. package/dist/cache/LRUCache.d.ts +93 -0
  46. package/dist/cache/LRUCache.d.ts.map +1 -0
  47. package/dist/cache/LRUCache.js +299 -0
  48. package/dist/cache/index.d.ts +1 -0
  49. package/dist/cache/index.d.ts.map +1 -1
  50. package/dist/cache/index.js +2 -1
  51. package/dist/collection/CollectionBrowser.d.ts +21 -1
  52. package/dist/collection/CollectionBrowser.d.ts.map +1 -1
  53. package/dist/collection/CollectionBrowser.js +130 -10
  54. package/dist/collection/CollectionIndexManager.d.ts +151 -0
  55. package/dist/collection/CollectionIndexManager.d.ts.map +1 -0
  56. package/dist/collection/CollectionIndexManager.js +499 -0
  57. package/dist/collection/CollectionSearch.d.ts +55 -0
  58. package/dist/collection/CollectionSearch.d.ts.map +1 -1
  59. package/dist/collection/CollectionSearch.js +338 -13
  60. package/dist/collection/CollectionSeeder.d.ts.map +1 -1
  61. package/dist/collection/CollectionSeeder.js +38 -1
  62. package/dist/collection/ElementInstaller.d.ts +31 -0
  63. package/dist/collection/ElementInstaller.d.ts.map +1 -1
  64. package/dist/collection/ElementInstaller.js +77 -15
  65. package/dist/collection/PersonaSubmitter.d.ts +1 -1
  66. package/dist/collection/PersonaSubmitter.d.ts.map +1 -1
  67. package/dist/collection/PersonaSubmitter.js +2 -2
  68. package/dist/collection/index.d.ts +1 -0
  69. package/dist/collection/index.d.ts.map +1 -1
  70. package/dist/collection/index.js +2 -1
  71. package/dist/config/ConfigManager.d.ts +78 -0
  72. package/dist/config/ConfigManager.d.ts.map +1 -0
  73. package/dist/config/ConfigManager.js +216 -0
  74. package/dist/config/element-types.d.ts +135 -0
  75. package/dist/config/element-types.d.ts.map +1 -0
  76. package/dist/config/element-types.js +108 -0
  77. package/dist/config/index.d.ts +2 -0
  78. package/dist/config/index.d.ts.map +1 -1
  79. package/dist/config/index.js +3 -1
  80. package/dist/config/portfolio-constants.d.ts +83 -0
  81. package/dist/config/portfolio-constants.d.ts.map +1 -0
  82. package/dist/config/portfolio-constants.js +99 -0
  83. package/dist/elements/BaseElement.d.ts +14 -2
  84. package/dist/elements/BaseElement.d.ts.map +1 -1
  85. package/dist/elements/BaseElement.js +88 -6
  86. package/dist/elements/agents/Agent.d.ts +10 -1
  87. package/dist/elements/agents/Agent.d.ts.map +1 -1
  88. package/dist/elements/agents/Agent.js +66 -19
  89. package/dist/elements/agents/AgentManager.d.ts +2 -0
  90. package/dist/elements/agents/AgentManager.d.ts.map +1 -1
  91. package/dist/elements/agents/AgentManager.js +12 -10
  92. package/dist/elements/skills/Skill.d.ts +10 -1
  93. package/dist/elements/skills/Skill.d.ts.map +1 -1
  94. package/dist/elements/skills/Skill.js +40 -3
  95. package/dist/elements/skills/SkillManager.d.ts +1 -0
  96. package/dist/elements/skills/SkillManager.d.ts.map +1 -1
  97. package/dist/elements/skills/SkillManager.js +10 -4
  98. package/dist/elements/templates/Template.d.ts +10 -1
  99. package/dist/elements/templates/Template.d.ts.map +1 -1
  100. package/dist/elements/templates/Template.js +35 -18
  101. package/dist/elements/templates/TemplateManager.d.ts +1 -1
  102. package/dist/elements/templates/TemplateManager.d.ts.map +1 -1
  103. package/dist/elements/templates/TemplateManager.js +6 -5
  104. package/dist/generated/version.d.ts +2 -2
  105. package/dist/generated/version.js +3 -3
  106. package/dist/index.barrel.d.ts +1 -2
  107. package/dist/index.barrel.d.ts.map +1 -1
  108. package/dist/index.barrel.js +2 -4
  109. package/dist/index.d.ts +143 -25
  110. package/dist/index.d.ts.map +1 -1
  111. package/dist/index.js +1883 -310
  112. package/dist/persona/PersonaElement.d.ts +10 -0
  113. package/dist/persona/PersonaElement.d.ts.map +1 -1
  114. package/dist/persona/PersonaElement.js +55 -32
  115. package/dist/persona/PersonaElementManager.d.ts.map +1 -1
  116. package/dist/persona/PersonaElementManager.js +13 -11
  117. package/dist/persona/PersonaLoader.d.ts.map +1 -1
  118. package/dist/persona/PersonaLoader.js +8 -2
  119. package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -1
  120. package/dist/persona/export-import/PersonaImporter.js +24 -5
  121. package/dist/persona/export-import/PersonaSharer.d.ts +21 -0
  122. package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -1
  123. package/dist/persona/export-import/PersonaSharer.js +198 -22
  124. package/dist/portfolio/DefaultElementProvider.d.ts +90 -0
  125. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -1
  126. package/dist/portfolio/DefaultElementProvider.js +499 -7
  127. package/dist/portfolio/GitHubPortfolioIndexer.d.ts +129 -0
  128. package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -0
  129. package/dist/portfolio/GitHubPortfolioIndexer.js +475 -0
  130. package/dist/portfolio/MigrationManager.d.ts.map +1 -1
  131. package/dist/portfolio/MigrationManager.js +136 -3
  132. package/dist/portfolio/PortfolioIndexManager.d.ts +130 -0
  133. package/dist/portfolio/PortfolioIndexManager.d.ts.map +1 -0
  134. package/dist/portfolio/PortfolioIndexManager.js +478 -0
  135. package/dist/portfolio/PortfolioManager.d.ts +5 -0
  136. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  137. package/dist/portfolio/PortfolioManager.js +61 -20
  138. package/dist/portfolio/PortfolioRepoManager.d.ts +75 -0
  139. package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -0
  140. package/dist/portfolio/PortfolioRepoManager.js +337 -0
  141. package/dist/portfolio/UnifiedIndexManager.d.ts +388 -0
  142. package/dist/portfolio/UnifiedIndexManager.d.ts.map +1 -0
  143. package/dist/portfolio/UnifiedIndexManager.js +1434 -0
  144. package/dist/portfolio/index.d.ts +15 -0
  145. package/dist/portfolio/index.d.ts.map +1 -0
  146. package/dist/portfolio/index.js +15 -0
  147. package/dist/portfolio/types.d.ts +7 -0
  148. package/dist/portfolio/types.d.ts.map +1 -1
  149. package/dist/portfolio/types.js +6 -1
  150. package/dist/security/InputValidator.d.ts.map +1 -1
  151. package/dist/security/InputValidator.js +50 -48
  152. package/dist/security/audit/SecurityAuditor.d.ts.map +1 -1
  153. package/dist/security/audit/SecurityAuditor.js +17 -9
  154. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  155. package/dist/security/audit/config/suppressions.js +19 -3
  156. package/dist/security/contentValidator.d.ts +2 -0
  157. package/dist/security/contentValidator.d.ts.map +1 -1
  158. package/dist/security/contentValidator.js +115 -4
  159. package/dist/security/secureYamlParser.d.ts +1 -0
  160. package/dist/security/secureYamlParser.d.ts.map +1 -1
  161. package/dist/security/secureYamlParser.js +29 -7
  162. package/dist/security/securityMonitor.d.ts +1 -1
  163. package/dist/security/securityMonitor.d.ts.map +1 -1
  164. package/dist/security/securityMonitor.js +1 -1
  165. package/dist/security/tokenManager.d.ts +1 -1
  166. package/dist/security/tokenManager.d.ts.map +1 -1
  167. package/dist/security/tokenManager.js +30 -10
  168. package/dist/server/ServerSetup.d.ts +22 -2
  169. package/dist/server/ServerSetup.d.ts.map +1 -1
  170. package/dist/server/ServerSetup.js +77 -12
  171. package/dist/server/tools/AuthTools.d.ts.map +1 -1
  172. package/dist/server/tools/AuthTools.js +33 -1
  173. package/dist/server/tools/BuildInfoTools.d.ts +25 -0
  174. package/dist/server/tools/BuildInfoTools.d.ts.map +1 -0
  175. package/dist/server/tools/BuildInfoTools.js +36 -0
  176. package/dist/server/tools/CollectionTools.d.ts.map +1 -1
  177. package/dist/server/tools/CollectionTools.js +55 -46
  178. package/dist/server/tools/ConfigTools.d.ts.map +1 -1
  179. package/dist/server/tools/ConfigTools.js +29 -1
  180. package/dist/server/tools/PersonaTools.d.ts +4 -2
  181. package/dist/server/tools/PersonaTools.d.ts.map +1 -1
  182. package/dist/server/tools/PersonaTools.js +5 -152
  183. package/dist/server/tools/PortfolioTools.d.ts +12 -0
  184. package/dist/server/tools/PortfolioTools.d.ts.map +1 -0
  185. package/dist/server/tools/PortfolioTools.js +221 -0
  186. package/dist/server/tools/index.d.ts +3 -1
  187. package/dist/server/tools/index.d.ts.map +1 -1
  188. package/dist/server/tools/index.js +4 -2
  189. package/dist/server/types.d.ts +40 -5
  190. package/dist/server/types.d.ts.map +1 -1
  191. package/dist/server/types.js +1 -1
  192. package/dist/services/BuildInfoService.d.ts +84 -0
  193. package/dist/services/BuildInfoService.d.ts.map +1 -0
  194. package/dist/services/BuildInfoService.js +271 -0
  195. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts +54 -0
  196. package/dist/tools/portfolio/PortfolioElementAdapter.d.ts.map +1 -0
  197. package/dist/tools/portfolio/PortfolioElementAdapter.js +229 -0
  198. package/dist/tools/portfolio/submitToPortfolioTool.d.ts +164 -0
  199. package/dist/tools/portfolio/submitToPortfolioTool.d.ts.map +1 -0
  200. package/dist/tools/portfolio/submitToPortfolioTool.js +1523 -0
  201. package/dist/tools/portfolio/types.d.ts +41 -0
  202. package/dist/tools/portfolio/types.d.ts.map +1 -0
  203. package/dist/tools/portfolio/types.js +15 -0
  204. package/dist/types/collection.d.ts +51 -0
  205. package/dist/types/collection.d.ts.map +1 -1
  206. package/dist/types/collection.js +1 -1
  207. package/dist/utils/EarlyTerminationSearch.d.ts +41 -0
  208. package/dist/utils/EarlyTerminationSearch.d.ts.map +1 -0
  209. package/dist/utils/EarlyTerminationSearch.js +164 -0
  210. package/dist/utils/ErrorHandler.d.ts +86 -0
  211. package/dist/utils/ErrorHandler.d.ts.map +1 -0
  212. package/dist/utils/ErrorHandler.js +201 -0
  213. package/dist/utils/FileDiscoveryUtil.d.ts +53 -0
  214. package/dist/utils/FileDiscoveryUtil.d.ts.map +1 -0
  215. package/dist/utils/FileDiscoveryUtil.js +169 -0
  216. package/dist/utils/GitHubRateLimiter.d.ts +88 -0
  217. package/dist/utils/GitHubRateLimiter.d.ts.map +1 -0
  218. package/dist/utils/GitHubRateLimiter.js +315 -0
  219. package/dist/utils/PerformanceMonitor.d.ts +134 -0
  220. package/dist/utils/PerformanceMonitor.d.ts.map +1 -0
  221. package/dist/utils/PerformanceMonitor.js +347 -0
  222. package/dist/utils/RateLimiter.d.ts.map +1 -0
  223. package/dist/utils/RateLimiter.js +172 -0
  224. package/dist/utils/SecureDownloader.d.ts +241 -0
  225. package/dist/utils/SecureDownloader.d.ts.map +1 -0
  226. package/dist/utils/SecureDownloader.js +759 -0
  227. package/dist/utils/ToolCache.d.ts +82 -0
  228. package/dist/utils/ToolCache.d.ts.map +1 -0
  229. package/dist/utils/ToolCache.js +196 -0
  230. package/dist/utils/errorCodes.d.ts +136 -0
  231. package/dist/utils/errorCodes.d.ts.map +1 -0
  232. package/dist/utils/errorCodes.js +87 -0
  233. package/dist/utils/index.d.ts +3 -0
  234. package/dist/utils/index.d.ts.map +1 -1
  235. package/dist/utils/index.js +4 -1
  236. package/dist/utils/installation.d.ts +1 -1
  237. package/dist/utils/installation.d.ts.map +1 -1
  238. package/dist/utils/installation.js +9 -8
  239. package/dist/utils/searchUtils.d.ts +31 -0
  240. package/dist/utils/searchUtils.d.ts.map +1 -1
  241. package/dist/utils/searchUtils.js +62 -1
  242. package/package.json +17 -7
  243. package/dist/config/updateConfig.d.ts +0 -84
  244. package/dist/config/updateConfig.d.ts.map +0 -1
  245. package/dist/config/updateConfig.js +0 -148
  246. package/dist/server/tools/UpdateTools.d.ts +0 -10
  247. package/dist/server/tools/UpdateTools.d.ts.map +0 -1
  248. package/dist/server/tools/UpdateTools.js +0 -85
  249. package/dist/update/BackupManager.d.ts +0 -63
  250. package/dist/update/BackupManager.d.ts.map +0 -1
  251. package/dist/update/BackupManager.js +0 -370
  252. package/dist/update/DependencyChecker.d.ts +0 -41
  253. package/dist/update/DependencyChecker.d.ts.map +0 -1
  254. package/dist/update/DependencyChecker.js +0 -132
  255. package/dist/update/RateLimiter.d.ts.map +0 -1
  256. package/dist/update/RateLimiter.js +0 -172
  257. package/dist/update/SignatureVerifier.d.ts +0 -71
  258. package/dist/update/SignatureVerifier.d.ts.map +0 -1
  259. package/dist/update/SignatureVerifier.js +0 -214
  260. package/dist/update/UpdateChecker.d.ts +0 -132
  261. package/dist/update/UpdateChecker.d.ts.map +0 -1
  262. package/dist/update/UpdateChecker.js +0 -506
  263. package/dist/update/UpdateManager.d.ts +0 -60
  264. package/dist/update/UpdateManager.d.ts.map +0 -1
  265. package/dist/update/UpdateManager.js +0 -730
  266. package/dist/update/VersionManager.d.ts +0 -31
  267. package/dist/update/VersionManager.d.ts.map +0 -1
  268. package/dist/update/VersionManager.js +0 -181
  269. package/dist/update/index.d.ts +0 -9
  270. package/dist/update/index.d.ts.map +0 -1
  271. package/dist/update/index.js +0 -9
  272. /package/dist/{update → utils}/RateLimiter.d.ts +0 -0
@@ -0,0 +1,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 */"]}
@@ -3,4 +3,6 @@
3
3
  */
4
4
  export * from './constants.js';
5
5
  export * from './indicator-config.js';
6
+ export * from './element-types.js';
7
+ export * from './portfolio-constants.js';
6
8
  //# sourceMappingURL=index.d.ts.map
@@ -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"}
@@ -3,4 +3,6 @@
3
3
  */
4
4
  export * from './constants.js';
5
5
  export * from './indicator-config.js';
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29uZmlnL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLHVCQUF1QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDZW50cmFsIGV4cG9ydCBwb2ludCBmb3IgY29uZmlndXJhdGlvblxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vY29uc3RhbnRzLmpzJztcbmV4cG9ydCAqIGZyb20gJy4vaW5kaWNhdG9yLWNvbmZpZy5qcyc7Il19
6
+ export * from './element-types.js';
7
+ export * from './portfolio-constants.js';
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29uZmlnL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxnQkFBZ0IsQ0FBQztBQUMvQixjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsb0JBQW9CLENBQUM7QUFDbkMsY0FBYywwQkFBMEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQ2VudHJhbCBleHBvcnQgcG9pbnQgZm9yIGNvbmZpZ3VyYXRpb25cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL2NvbnN0YW50cy5qcyc7XG5leHBvcnQgKiBmcm9tICcuL2luZGljYXRvci1jb25maWcuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9lbGVtZW50LXR5cGVzLmpzJztcbmV4cG9ydCAqIGZyb20gJy4vcG9ydGZvbGlvLWNvbnN0YW50cy5qcyc7Il19