@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
@@ -1,730 +0,0 @@
1
- /**
2
- * Manage server updates and rollbacks
3
- */
4
- import * as path from 'path';
5
- import * as fs from 'fs/promises';
6
- import { safeExec } from '../utils/git.js';
7
- import { VersionManager } from './VersionManager.js';
8
- import { UpdateChecker } from './UpdateChecker.js';
9
- import { DependencyChecker } from './DependencyChecker.js';
10
- import { BackupManager } from './BackupManager.js';
11
- import { InstallationDetector } from '../utils/installation.js';
12
- import { logger } from '../utils/logger.js';
13
- import { compareVersions } from '../utils/version.js';
14
- import { FileOperations } from '../utils/fileOperations.js';
15
- import { UpdateConfigManager } from '../config/updateConfig.js';
16
- export class UpdateManager {
17
- versionManager;
18
- updateChecker;
19
- dependencyChecker;
20
- backupManager;
21
- rootDir;
22
- constructor(rootDir) {
23
- this.rootDir = rootDir || process.cwd();
24
- this.versionManager = new VersionManager();
25
- this.updateChecker = new UpdateChecker(this.versionManager);
26
- this.dependencyChecker = new DependencyChecker(this.versionManager);
27
- this.backupManager = new BackupManager(this.rootDir);
28
- }
29
- /**
30
- * Check for available updates
31
- */
32
- async checkForUpdates() {
33
- try {
34
- const result = await this.updateChecker.checkForUpdates();
35
- const text = this.updateChecker.formatUpdateCheckResult(result);
36
- return { text };
37
- }
38
- catch (error) {
39
- const text = this.updateChecker.formatUpdateCheckResult(null, error);
40
- return { text };
41
- }
42
- }
43
- /**
44
- * Perform server update
45
- */
46
- async updateServer(createBackup = true, personaIndicator = '') {
47
- const progress = [];
48
- try {
49
- // Detect installation type
50
- const installationType = InstallationDetector.getInstallationType();
51
- logger.info(`[UpdateManager] Detected installation type: ${installationType}`);
52
- // Handle npm installations differently
53
- if (installationType === 'npm') {
54
- return this.updateNpmInstallation(createBackup, personaIndicator);
55
- }
56
- // For git installations, proceed with existing logic
57
- // Step 1: Check dependencies
58
- progress.push({ step: 'dependencies', message: 'Checking system dependencies...', isComplete: false });
59
- const dependencies = await this.dependencyChecker.checkDependencies();
60
- if (!dependencies.git.installed || dependencies.git.error) {
61
- return {
62
- text: personaIndicator + '❌ **Update Failed**\n\n' +
63
- 'Git is required for updates but is not available.\n' +
64
- dependencies.git.error || 'Git is not installed.'
65
- };
66
- }
67
- if (!dependencies.npm.installed || dependencies.npm.error) {
68
- return {
69
- text: personaIndicator + '❌ **Update Failed**\n\n' +
70
- 'npm is required for updates but is not available.\n' +
71
- dependencies.npm.error || 'npm is not installed.'
72
- };
73
- }
74
- progress[0].isComplete = true;
75
- // Step 2: Create backup if requested
76
- if (createBackup) {
77
- progress.push({ step: 'backup', message: 'Creating backup...', isComplete: false });
78
- const currentVersion = await this.versionManager.getCurrentVersion();
79
- const backup = await this.backupManager.createBackup(currentVersion);
80
- progress[1].isComplete = true;
81
- progress[1].message = `Backup created at: ${backup.timestamp}`;
82
- }
83
- // Step 3: Git fetch
84
- progress.push({ step: 'fetch', message: 'Fetching latest changes...', isComplete: false });
85
- await safeExec('git', ['fetch', 'origin'], { cwd: this.rootDir });
86
- progress[progress.length - 1].isComplete = true;
87
- // Step 4: Check for uncommitted changes
88
- progress.push({ step: 'check', message: 'Checking for uncommitted changes...', isComplete: false });
89
- const { stdout: statusOutput } = await safeExec('git', ['status', '--porcelain'], { cwd: this.rootDir });
90
- if (statusOutput.trim()) {
91
- return {
92
- text: personaIndicator + '❌ **Update Failed**\n\n' +
93
- 'You have uncommitted changes. Please commit or stash them before updating.\n\n' +
94
- 'Modified files:\n' + statusOutput
95
- };
96
- }
97
- progress[progress.length - 1].isComplete = true;
98
- // Step 5: Git pull
99
- progress.push({ step: 'pull', message: 'Pulling latest changes...', isComplete: false });
100
- const { stdout: pullOutput } = await safeExec('git', ['pull', 'origin', 'main'], { cwd: this.rootDir });
101
- progress[progress.length - 1].isComplete = true;
102
- // Check if already up to date
103
- if (pullOutput.includes('Already up to date')) {
104
- return {
105
- text: personaIndicator + '✅ **Already Up to Date**\n\n' +
106
- 'Your DollhouseMCP installation is already at the latest version.\n\n' +
107
- 'No changes were pulled from the repository.'
108
- };
109
- }
110
- // Step 6: npm install
111
- progress.push({ step: 'install', message: 'Installing dependencies...', isComplete: false });
112
- await safeExec('npm', ['install'], { cwd: this.rootDir });
113
- progress[progress.length - 1].isComplete = true;
114
- // Step 7: Build
115
- progress.push({ step: 'build', message: 'Building TypeScript...', isComplete: false });
116
- await safeExec('npm', ['run', 'build'], { cwd: this.rootDir });
117
- progress[progress.length - 1].isComplete = true;
118
- // Step 8: Cleanup old backups
119
- if (createBackup) {
120
- progress.push({ step: 'cleanup', message: 'Cleaning up old backups...', isComplete: false });
121
- const deletedCount = await this.backupManager.cleanupOldBackups();
122
- progress[progress.length - 1].isComplete = true;
123
- progress[progress.length - 1].message = `Cleaned up ${deletedCount} old backup(s)`;
124
- }
125
- // Format success message
126
- const successParts = [
127
- personaIndicator + '✅ **Update Complete!**\n\n',
128
- '**Update Summary:**\n'
129
- ];
130
- progress.forEach(p => {
131
- successParts.push(`${p.isComplete ? '✅' : '❌'} ${p.message}\n`);
132
- });
133
- successParts.push('\n**Next Steps:**\n', '1. The server will restart automatically\n', '2. All personas will be reloaded\n', '3. Check `get_server_status` to verify the new version\n\n', '💡 **Tip:** If you encounter issues, use `rollback_update true` to restore the previous version.');
134
- return { text: successParts.join('') };
135
- }
136
- catch (error) {
137
- const errorMessage = error instanceof Error ? error.message : String(error);
138
- return {
139
- text: personaIndicator + '❌ **Update Failed**\n\n' +
140
- 'Error: ' + errorMessage + '\n\n' +
141
- '**Progress:**\n' +
142
- progress.map(p => `${p.isComplete ? '✅' : '❌'} ${p.message}`).join('\n') + '\n\n' +
143
- '**Recovery Options:**\n' +
144
- '• Try running the update again\n' +
145
- '• Check your internet connection\n' +
146
- '• Ensure you have proper permissions\n' +
147
- '• If a backup was created, use `rollback_update true` to restore'
148
- };
149
- }
150
- }
151
- /**
152
- * Rollback to previous version
153
- */
154
- async rollbackUpdate(force = false, personaIndicator = '') {
155
- try {
156
- // Check installation type
157
- const installationType = InstallationDetector.getInstallationType();
158
- if (installationType === 'npm') {
159
- return this.rollbackNpmInstallation(force, personaIndicator);
160
- }
161
- // For git installations, use existing logic
162
- // Get latest backup
163
- const latestBackup = await this.backupManager.getLatestBackup();
164
- if (!latestBackup) {
165
- return {
166
- text: personaIndicator + '❌ **No Backups Found**\n\n' +
167
- 'There are no backups available to restore.\n\n' +
168
- 'Backups are created automatically when you run `update_server true`.'
169
- };
170
- }
171
- // Check if rollback is needed
172
- if (!force) {
173
- try {
174
- // Test if the server is working by checking version
175
- await this.versionManager.getCurrentVersion();
176
- return {
177
- text: personaIndicator + '⚠️ **Rollback Confirmation Required**\n\n' +
178
- 'The server appears to be working normally.\n\n' +
179
- `**Latest Backup:** ${latestBackup.timestamp}\n` +
180
- `**Backup Version:** ${latestBackup.version || 'Unknown'}\n\n` +
181
- 'To force rollback anyway, use: `rollback_update true`\n\n' +
182
- '⚠️ **Warning:** This will restore all files to the backup state.'
183
- };
184
- }
185
- catch {
186
- // Server is broken, proceed with rollback
187
- }
188
- }
189
- // Perform rollback
190
- await this.backupManager.restoreBackup(latestBackup.path);
191
- // Reinstall dependencies
192
- await safeExec('npm', ['install'], { cwd: this.rootDir });
193
- // Rebuild
194
- await safeExec('npm', ['run', 'build'], { cwd: this.rootDir });
195
- return {
196
- text: personaIndicator + '✅ **Rollback Complete!**\n\n' +
197
- `Restored from backup: ${latestBackup.timestamp}\n` +
198
- `Backup version: ${latestBackup.version || 'Unknown'}\n\n` +
199
- '**What was restored:**\n' +
200
- '• All source files\n' +
201
- '• Configuration files\n' +
202
- '• Dependencies reinstalled\n' +
203
- '• TypeScript rebuilt\n\n' +
204
- '**Next Steps:**\n' +
205
- '1. The server will restart automatically\n' +
206
- '2. Check `get_server_status` to verify the version\n' +
207
- '3. Test your personas to ensure everything works'
208
- };
209
- }
210
- catch (error) {
211
- const errorMessage = error instanceof Error ? error.message : String(error);
212
- return {
213
- text: personaIndicator + '❌ **Rollback Failed**\n\n' +
214
- 'Error: ' + errorMessage + '\n\n' +
215
- '**Manual Recovery:**\n' +
216
- '1. Check the backups directory: ../dollhousemcp-backups/\n' +
217
- '2. Manually restore files if needed\n' +
218
- '3. Run `npm install` and `npm run build`\n' +
219
- '4. Contact support if issues persist'
220
- };
221
- }
222
- }
223
- /**
224
- * Update npm installation
225
- */
226
- async updateNpmInstallation(createBackup, personaIndicator = '') {
227
- try {
228
- logger.info('[UpdateManager] Starting npm update process');
229
- // Check npm is available
230
- const dependencies = await this.dependencyChecker.checkDependencies();
231
- if (!dependencies.npm.installed || dependencies.npm.error) {
232
- return {
233
- text: personaIndicator + '❌ **Update Failed**\n\n' +
234
- 'npm is required for updates but is not available.\n' +
235
- dependencies.npm.error || 'npm is not installed.'
236
- };
237
- }
238
- // Get current version
239
- const currentVersion = await this.versionManager.getCurrentVersion();
240
- logger.info(`[UpdateManager] Current version: ${currentVersion}`);
241
- // Check latest version from npm registry
242
- logger.info('[UpdateManager] Checking npm registry for latest version');
243
- // Security: Validate package name to prevent any potential injection
244
- const packageName = '@dollhousemcp/mcp-server';
245
- if (!/^@[a-z0-9-]+\/[a-z0-9-]+$/.test(packageName)) {
246
- throw new Error('Invalid package name format');
247
- }
248
- const { stdout: npmViewOutput } = await safeExec('npm', ['view', packageName, 'version'], {
249
- cwd: this.rootDir,
250
- timeout: 30000
251
- });
252
- const latestVersion = npmViewOutput.trim();
253
- logger.info(`[UpdateManager] Latest npm version: ${latestVersion}`);
254
- // Compare versions
255
- const comparison = compareVersions(currentVersion, latestVersion);
256
- if (comparison >= 0) {
257
- return {
258
- text: personaIndicator + '✅ **Already up to date!**\n\n' +
259
- `Current version: ${currentVersion}\n` +
260
- `Latest version: ${latestVersion}\n\n` +
261
- 'No update needed.'
262
- };
263
- }
264
- // Get configuration
265
- const config = UpdateConfigManager.getInstance();
266
- // Progress tracking
267
- let progressMessage = personaIndicator + '🔄 **NPM Update in Progress**\n\n';
268
- progressMessage += '**Steps:**\n';
269
- progressMessage += '✅ Version check complete\n';
270
- progressMessage += '⏳ Creating backup...\n';
271
- // For npm installations, backup is mandatory for safety
272
- logger.info('[UpdateManager] Creating backup before npm update');
273
- try {
274
- // For npm installations, we backup the global installation directory
275
- const npmGlobalPath = InstallationDetector.getNpmGlobalPath();
276
- if (!npmGlobalPath) {
277
- throw new Error('Could not determine npm global installation path');
278
- }
279
- // Create npm-specific backup with progress
280
- const backupPath = await this.backupManager.createNpmBackup(npmGlobalPath, currentVersion);
281
- logger.info(`[UpdateManager] Backup created at: ${backupPath}`);
282
- progressMessage = progressMessage.replace('⏳ Creating backup...', '✅ Backup created');
283
- progressMessage += '⏳ Downloading and installing update...\n';
284
- }
285
- catch (backupError) {
286
- logger.error('[UpdateManager] Backup failed:', backupError);
287
- return {
288
- text: personaIndicator + '❌ **Update Failed**\n\n' +
289
- 'Failed to create backup before update.\n' +
290
- 'Error: ' + (backupError instanceof Error ? backupError.message : String(backupError)) + '\n\n' +
291
- '**Note:** Backup is mandatory for npm installations to ensure safe rollback.\n' +
292
- 'Please check disk space and permissions.'
293
- };
294
- }
295
- // Perform npm update with progress
296
- logger.info('[UpdateManager] Running npm update -g @dollhousemcp/mcp-server');
297
- progressMessage += '\n**Progress:**\n';
298
- progressMessage += '```\n';
299
- progressMessage += 'Running: npm update -g @dollhousemcp/mcp-server\n';
300
- progressMessage += 'This may take a few minutes...\n';
301
- progressMessage += '```\n';
302
- const updateResult = await safeExec('npm', ['update', '-g', '@dollhousemcp/mcp-server'], {
303
- cwd: this.rootDir,
304
- timeout: config.getNpmUpdateTimeout()
305
- });
306
- logger.info('[UpdateManager] npm update completed', updateResult);
307
- // Verify update succeeded
308
- const { stdout: verifyOutput } = await safeExec('npm', ['list', '-g', '@dollhousemcp/mcp-server', '--depth=0'], {
309
- cwd: this.rootDir,
310
- timeout: 30000
311
- });
312
- const versionMatch = verifyOutput.match(/@dollhousemcp\/mcp-server@(\d+\.\d+\.\d+)/);
313
- const installedVersion = versionMatch ? versionMatch[1] : 'unknown';
314
- if (installedVersion !== latestVersion) {
315
- logger.warn(`[UpdateManager] Version mismatch after update. Expected: ${latestVersion}, Got: ${installedVersion}`);
316
- }
317
- return {
318
- text: personaIndicator + '✅ **Update Complete!**\n\n' +
319
- `Updated from v${currentVersion} to v${latestVersion}\n\n` +
320
- '**What was updated:**\n' +
321
- '• DollhouseMCP server package\n' +
322
- '• All dependencies\n\n' +
323
- '**Next Steps:**\n' +
324
- '1. The server will restart automatically\n' +
325
- '2. Check `get_server_status` to verify the new version\n' +
326
- '3. Test your personas to ensure everything works\n\n' +
327
- '💡 **Tip:** If you encounter issues, use `rollback_update true` to restore the previous version.'
328
- };
329
- }
330
- catch (error) {
331
- logger.error('[UpdateManager] npm update failed:', error);
332
- const errorMessage = error instanceof Error ? error.message : String(error);
333
- return {
334
- text: personaIndicator + '❌ **Update Failed**\n\n' +
335
- 'Error: ' + errorMessage + '\n\n' +
336
- '**Troubleshooting:**\n' +
337
- '1. Ensure you have permission to update global npm packages\n' +
338
- '2. Try running with sudo if on macOS/Linux\n' +
339
- '3. Check your internet connection\n' +
340
- '4. Verify npm registry is accessible\n\n' +
341
- '**Manual Update:**\n' +
342
- '```\n' +
343
- 'npm update -g @dollhousemcp/mcp-server\n' +
344
- '```'
345
- };
346
- }
347
- }
348
- /**
349
- * Rollback npm installation
350
- */
351
- async rollbackNpmInstallation(force, personaIndicator = '') {
352
- try {
353
- logger.info('[UpdateManager] Starting npm rollback process');
354
- // Get npm backup manifest from configuration
355
- const config = UpdateConfigManager.getInstance();
356
- const npmBackupsDir = config.getNpmBackupDir();
357
- const manifestPath = path.join(npmBackupsDir, 'manifest.json');
358
- let manifest;
359
- try {
360
- const content = await fs.readFile(manifestPath, 'utf-8');
361
- manifest = JSON.parse(content);
362
- }
363
- catch (error) {
364
- return {
365
- text: personaIndicator + '❌ **No NPM Backups Found**\n\n' +
366
- 'There are no npm backups available to restore.\n\n' +
367
- 'Backups are created automatically when you run `update_server true` with npm installations.'
368
- };
369
- }
370
- if (!manifest.backups || manifest.backups.length === 0) {
371
- return {
372
- text: personaIndicator + '❌ **No NPM Backups Found**\n\n' +
373
- 'The backup manifest is empty.\n\n' +
374
- 'Backups are created automatically when you run `update_server true`.'
375
- };
376
- }
377
- // Get latest backup
378
- const latestBackup = manifest.backups[0];
379
- // Check if rollback is needed
380
- if (!force) {
381
- try {
382
- // Test if the server is working
383
- await this.versionManager.getCurrentVersion();
384
- return {
385
- text: personaIndicator + '⚠️ **Rollback Confirmation Required**\n\n' +
386
- 'The server appears to be working normally.\n\n' +
387
- `**Latest Backup:** ${latestBackup.timestamp}\n` +
388
- `**Backup Version:** ${latestBackup.version || 'Unknown'}\n\n` +
389
- 'To force rollback anyway, use: `rollback_update true`\n\n' +
390
- '⚠️ **Warning:** This will restore the npm package to the backup state.'
391
- };
392
- }
393
- catch {
394
- // Server is broken, proceed with rollback
395
- }
396
- }
397
- // Get npm global path
398
- const npmGlobalPath = InstallationDetector.getNpmGlobalPath();
399
- if (!npmGlobalPath) {
400
- return {
401
- text: personaIndicator + '❌ **Rollback Failed**\n\n' +
402
- 'Could not determine npm global installation path.\n\n' +
403
- 'Please reinstall manually:\n' +
404
- '```\n' +
405
- 'npm install -g @dollhousemcp/mcp-server@' + (latestBackup.version || 'latest') + '\n' +
406
- '```'
407
- };
408
- }
409
- logger.info(`[UpdateManager] Restoring npm backup from: ${latestBackup.path}`);
410
- // First validate that the backup is restorable
411
- const backupPackagePath = path.join(latestBackup.path, 'package');
412
- try {
413
- await fs.access(backupPackagePath);
414
- const packageJsonPath = path.join(backupPackagePath, 'package.json');
415
- await fs.access(packageJsonPath);
416
- }
417
- catch (error) {
418
- return {
419
- text: personaIndicator + '❌ **Backup Validation Failed**\n\n' +
420
- 'The backup appears to be corrupted or incomplete.\n\n' +
421
- '**Details:**\n' +
422
- `Backup path: ${latestBackup.path}\n` +
423
- `Error: ${error instanceof Error ? error.message : String(error)}\n\n` +
424
- '**Manual Recovery:**\n' +
425
- 'Try another backup or reinstall manually:\n' +
426
- '```bash\n' +
427
- 'npm install -g @dollhousemcp/mcp-server@' + (latestBackup.version || '1.4.0') + '\n' +
428
- '```'
429
- };
430
- }
431
- // Use transaction for atomic operations
432
- const tempPath = `${npmGlobalPath}.tmp-${Date.now()}`;
433
- const backupPath = `${npmGlobalPath}.backup-${Date.now()}`;
434
- const transaction = FileOperations.createTransaction();
435
- try {
436
- // Step 1: Copy backup to temporary location
437
- await transaction.addCopy(backupPackagePath, tempPath);
438
- // Step 2: Move current installation to backup (atomic)
439
- await transaction.addMove(npmGlobalPath, backupPath);
440
- // Step 3: Move temp to final location (atomic)
441
- await transaction.addMove(tempPath, npmGlobalPath);
442
- // All operations successful, commit the transaction
443
- transaction.commit();
444
- // Step 4: Clean up old backup (not part of transaction)
445
- await fs.rm(backupPath, { recursive: true, force: true }).catch(() => {
446
- logger.warn(`[UpdateManager] Failed to cleanup backup at: ${backupPath}`);
447
- });
448
- }
449
- catch (rollbackError) {
450
- logger.error('[UpdateManager] Rollback operation failed, attempting recovery:', rollbackError);
451
- // Rollback all operations
452
- if (transaction.hasOperations()) {
453
- await transaction.rollback();
454
- }
455
- // Additional recovery attempt
456
- try {
457
- // Check if npm path exists and is accessible
458
- await fs.access(npmGlobalPath);
459
- logger.info('[UpdateManager] NPM installation appears intact after rollback');
460
- }
461
- catch {
462
- // Try to restore from backup if main path is missing
463
- try {
464
- await fs.rename(backupPath, npmGlobalPath);
465
- logger.info('[UpdateManager] Restored from backup path');
466
- }
467
- catch {
468
- logger.error('[UpdateManager] Complete rollback failure - manual intervention required');
469
- }
470
- }
471
- throw rollbackError;
472
- }
473
- return {
474
- text: personaIndicator + '✅ **NPM Rollback Complete!**\n\n' +
475
- `Restored from backup: ${latestBackup.timestamp}\n` +
476
- `Backup version: ${latestBackup.version || 'Unknown'}\n\n` +
477
- '**What was restored:**\n' +
478
- '• NPM package files\n' +
479
- '• All dependencies\n\n' +
480
- '**Next Steps:**\n' +
481
- '1. The server will restart automatically\n' +
482
- '2. Check `get_server_status` to verify the version\n' +
483
- '3. Test your personas to ensure everything works'
484
- };
485
- }
486
- catch (error) {
487
- logger.error('[UpdateManager] npm rollback failed:', error);
488
- const errorMessage = error instanceof Error ? error.message : String(error);
489
- return {
490
- text: personaIndicator + '❌ **NPM Rollback Failed**\n\n' +
491
- 'Error: ' + errorMessage + '\n\n' +
492
- '**Manual Recovery:**\n' +
493
- '1. Check the backups directory: ~/.dollhouse/backups/npm/\n' +
494
- '2. Reinstall a specific version:\n' +
495
- ' ```\n' +
496
- ' npm install -g @dollhousemcp/mcp-server@1.4.0\n' +
497
- ' ```\n' +
498
- '3. Contact support if issues persist'
499
- };
500
- }
501
- }
502
- /**
503
- * Copy directory recursively with progress reporting
504
- * @deprecated Use FileOperations.copyDirectory instead
505
- */
506
- async copyDirectory(src, dest) {
507
- await FileOperations.copyDirectory(src, dest, {
508
- excludePatterns: ['.git', 'node_modules'],
509
- onProgress: (copied, total, file) => {
510
- logger.debug(`[UpdateManager] Copying files: ${copied}/${total} - ${path.basename(file)}`);
511
- }
512
- });
513
- }
514
- /**
515
- * Convert npm installation to git installation
516
- */
517
- async convertToGitInstallation(targetDir, confirm = false, personaIndicator = '') {
518
- try {
519
- const installationType = InstallationDetector.getInstallationType();
520
- if (installationType === 'git') {
521
- return {
522
- text: personaIndicator + '⚠️ **Already a Git Installation**\n\n' +
523
- 'This server is already running from a git installation.\n' +
524
- 'No conversion needed.'
525
- };
526
- }
527
- if (installationType === 'unknown') {
528
- return {
529
- text: personaIndicator + '❌ **Installation Type Unknown**\n\n' +
530
- 'Cannot determine the current installation type.\n' +
531
- 'Please check your installation manually.'
532
- };
533
- }
534
- // Default target directory
535
- const defaultTargetDir = path.join(process.env.HOME || '', '.dollhouse', 'mcp-server-git');
536
- const gitTargetDir = targetDir || defaultTargetDir;
537
- if (!confirm) {
538
- return {
539
- text: personaIndicator + '🔄 **Convert to Git Installation**\n\n' +
540
- '**This will:**\n' +
541
- `1. Clone DollhouseMCP to: ${gitTargetDir}\n` +
542
- '2. Copy your portfolio and settings\n' +
543
- '3. Build the TypeScript code\n' +
544
- '4. Provide Claude Desktop configuration\n\n' +
545
- '**Benefits of Git Installation:**\n' +
546
- '• Full control over updates\n' +
547
- '• Access to development branches\n' +
548
- '• Ability to contribute changes\n' +
549
- '• Rollback to any commit\n\n' +
550
- '**To proceed:**\n' +
551
- '`convert_to_git_installation true`\n\n' +
552
- '**To use custom directory:**\n' +
553
- '`convert_to_git_installation "/path/to/dir" true`'
554
- };
555
- }
556
- logger.info(`[UpdateManager] Starting conversion to git installation at: ${gitTargetDir}`);
557
- // Check if target directory already exists
558
- try {
559
- await fs.access(gitTargetDir);
560
- return {
561
- text: personaIndicator + '❌ **Target Directory Exists**\n\n' +
562
- `The directory ${gitTargetDir} already exists.\n\n` +
563
- '**Options:**\n' +
564
- '1. Remove the existing directory first\n' +
565
- '2. Choose a different target directory\n' +
566
- '3. Use the existing git installation'
567
- };
568
- }
569
- catch {
570
- // Directory doesn't exist, good to proceed
571
- }
572
- // Progress message builder
573
- let progressSteps = personaIndicator + '🔄 **Git Installation Progress**\n\n';
574
- progressSteps += '**Steps:**\n';
575
- progressSteps += '⏳ Cloning repository...\n';
576
- progressSteps += '⏳ Installing dependencies...\n';
577
- progressSteps += '⏳ Building TypeScript...\n';
578
- progressSteps += '⏳ Setting up configuration...\n\n';
579
- progressSteps += '**Current Step:** Cloning repository\n';
580
- progressSteps += '```\n';
581
- progressSteps += `Target: ${gitTargetDir}\n`;
582
- progressSteps += 'This may take a few minutes depending on your connection...\n';
583
- progressSteps += '```\n';
584
- // Get configuration
585
- const config = UpdateConfigManager.getInstance();
586
- // Step 1: Clone the repository
587
- logger.info('[UpdateManager] Cloning repository...');
588
- // SECURITY FIX: Validate gitTargetDir to prevent command injection
589
- // Previously: gitTargetDir passed directly to git clone command
590
- // Now: Reject paths starting with '--' to prevent git option injection
591
- if (gitTargetDir.startsWith('--')) {
592
- throw new Error('Invalid target directory: cannot start with git options');
593
- }
594
- await safeExec('git', ['clone', 'https://github.com/DollhouseMCP/mcp-server.git', gitTargetDir], {
595
- timeout: config.getGitCloneTimeout()
596
- });
597
- progressSteps = progressSteps.replace('⏳ Cloning repository...', '✅ Repository cloned');
598
- progressSteps = progressSteps.replace('**Current Step:** Cloning repository', '**Current Step:** Installing dependencies');
599
- // Step 2: Install dependencies
600
- logger.info('[UpdateManager] Installing dependencies...');
601
- await safeExec('npm', ['install'], {
602
- cwd: gitTargetDir,
603
- timeout: config.getNpmInstallTimeout()
604
- });
605
- progressSteps = progressSteps.replace('⏳ Installing dependencies...', '✅ Dependencies installed');
606
- progressSteps = progressSteps.replace('**Current Step:** Installing dependencies', '**Current Step:** Building TypeScript');
607
- // Step 3: Build TypeScript
608
- logger.info('[UpdateManager] Building TypeScript...');
609
- await safeExec('npm', ['run', 'build'], {
610
- cwd: gitTargetDir,
611
- timeout: config.getBuildTimeout()
612
- });
613
- progressSteps = progressSteps.replace('⏳ Building TypeScript...', '✅ TypeScript built');
614
- progressSteps = progressSteps.replace('**Current Step:** Building TypeScript', '**Current Step:** Configuration');
615
- // Step 4: Copy portfolio
616
- const portfolioSource = path.join(process.env.HOME || '', '.dollhouse', 'portfolio');
617
- const portfolioTarget = path.join(process.env.HOME || '', '.dollhouse', 'portfolio');
618
- logger.info('[UpdateManager] Portfolio will remain at: ' + portfolioTarget);
619
- // Step 5: Generate Claude Desktop config
620
- const configPath = path.join(gitTargetDir, 'dist', 'index.js');
621
- const claudeConfig = {
622
- mcpServers: {
623
- dollhousemcp: {
624
- command: 'node',
625
- args: [configPath]
626
- }
627
- }
628
- };
629
- return {
630
- text: personaIndicator + '✅ **Git Installation Complete!**\n\n' +
631
- `**Installation Location:** ${gitTargetDir}\n\n` +
632
- '**Next Steps:**\n\n' +
633
- '1. **Update Claude Desktop configuration:**\n' +
634
- ' ```json\n' +
635
- JSON.stringify(claudeConfig, null, 2) + '\n' +
636
- ' ```\n\n' +
637
- '2. **Restart Claude Desktop**\n\n' +
638
- '3. **Verify installation:**\n' +
639
- ' After restart, run `get_server_status` to confirm\n\n' +
640
- '**Your portfolio remains at:**\n' +
641
- ` ${portfolioTarget}\n\n` +
642
- '**To update in the future:**\n' +
643
- ' ```bash\n' +
644
- ` cd ${gitTargetDir}\n` +
645
- ' git pull\n' +
646
- ' npm install\n' +
647
- ' npm run build\n' +
648
- ' ```\n\n' +
649
- '💡 **Tip:** You can now use `update_server` to update via git!'
650
- };
651
- }
652
- catch (error) {
653
- logger.error('[UpdateManager] Git conversion failed:', error);
654
- const errorMessage = error instanceof Error ? error.message : String(error);
655
- return {
656
- text: personaIndicator + '❌ **Conversion Failed**\n\n' +
657
- 'Error: ' + errorMessage + '\n\n' +
658
- '**Troubleshooting:**\n' +
659
- '1. Ensure git is installed\n' +
660
- '2. Check internet connection\n' +
661
- '3. Verify you have write permissions\n' +
662
- '4. Try a different target directory'
663
- };
664
- }
665
- }
666
- /**
667
- * Get current server status
668
- */
669
- async getServerStatus(personaIndicator = '') {
670
- try {
671
- const currentVersion = await this.versionManager.getCurrentVersion();
672
- const dependencies = await this.dependencyChecker.checkDependencies();
673
- const backups = await this.backupManager.listBackups();
674
- const rateLimitStatus = this.updateChecker.getRateLimitStatus();
675
- // Get installation type
676
- const installationType = InstallationDetector.getInstallationType();
677
- const installationDesc = InstallationDetector.getInstallationDescription();
678
- // Get git status (only for git installations)
679
- let gitStatus = 'N/A';
680
- let gitBranch = 'N/A';
681
- let lastCommit = 'N/A';
682
- if (installationType === 'git') {
683
- try {
684
- const { stdout: branchOutput } = await safeExec('git', ['branch', '--show-current'], { cwd: this.rootDir });
685
- gitBranch = branchOutput.trim() || 'detached';
686
- const { stdout: statusOutput } = await safeExec('git', ['status', '--porcelain'], { cwd: this.rootDir });
687
- gitStatus = statusOutput.trim() ? 'Modified' : 'Clean';
688
- const { stdout: logOutput } = await safeExec('git', ['log', '-1', '--oneline'], { cwd: this.rootDir });
689
- lastCommit = logOutput.trim();
690
- }
691
- catch {
692
- // Git commands failed, use defaults
693
- }
694
- }
695
- const statusParts = [
696
- personaIndicator + '📊 **DollhouseMCP Server Status**\n\n',
697
- '**Version Information:**\n',
698
- `• Current Version: ${currentVersion}\n`,
699
- `• Installation Type: ${installationType} (${installationDesc})\n`,
700
- `• Git Branch: ${gitBranch}\n`,
701
- `• Git Status: ${gitStatus}\n`,
702
- `• Last Commit: ${lastCommit}\n\n`,
703
- '**Dependencies:**\n',
704
- this.dependencyChecker.formatDependencyStatus(dependencies),
705
- '\n\n**Backups:**\n',
706
- `• Total Backups: ${backups.length}\n`
707
- ];
708
- if (backups.length > 0) {
709
- statusParts.push(`• Latest Backup: ${backups[0].timestamp} (v${backups[0].version || 'unknown'})\n`);
710
- statusParts.push(`• Oldest Backup: ${backups[backups.length - 1].timestamp}\n`);
711
- }
712
- statusParts.push('\n**Rate Limit Status:**\n', `• Update Checks Remaining: ${rateLimitStatus.remainingRequests}/10 per hour\n`, `• Rate Limit Resets: ${rateLimitStatus.resetTime.toLocaleTimeString()}\n`);
713
- if (!rateLimitStatus.allowed && rateLimitStatus.waitTimeSeconds) {
714
- statusParts.push(`• ⏳ Wait ${rateLimitStatus.waitTimeSeconds} seconds before next check\n`);
715
- }
716
- statusParts.push('\n**Available Commands:**\n', '• `check_for_updates` - Check for new versions\n', '• `update_server true` - Update to latest version\n', '• `rollback_update true` - Restore from backup\n');
717
- return { text: statusParts.join('') };
718
- }
719
- catch (error) {
720
- const errorMessage = error instanceof Error ? error.message : String(error);
721
- return {
722
- text: personaIndicator + '❌ **Status Check Failed**\n\n' +
723
- 'Error: ' + errorMessage + '\n\n' +
724
- 'The server may be in an inconsistent state.\n' +
725
- 'Try running `update_server true` to fix issues.'
726
- };
727
- }
728
- }
729
- }
730
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"UpdateManager.js","sourceRoot":"","sources":["../../src/update/UpdateManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Manage server updates and rollbacks\n */\n\nimport * as path from 'path';\nimport * as fs from 'fs/promises';\nimport { safeExec } from '../utils/git.js';\nimport { VersionManager } from './VersionManager.js';\nimport { UpdateChecker } from './UpdateChecker.js';\nimport { DependencyChecker } from './DependencyChecker.js';\nimport { BackupManager } from './BackupManager.js';\nimport { InstallationDetector } from '../utils/installation.js';\nimport { logger } from '../utils/logger.js';\nimport { compareVersions } from '../utils/version.js';\nimport { FileOperations, FileTransaction } from '../utils/fileOperations.js';\nimport { UpdateConfigManager } from '../config/updateConfig.js';\n\nexport interface UpdateProgress {\n  step: string;\n  message: string;\n  isComplete: boolean;\n}\n\nexport class UpdateManager {\n  private versionManager: VersionManager;\n  private updateChecker: UpdateChecker;\n  private dependencyChecker: DependencyChecker;\n  private backupManager: BackupManager;\n  private rootDir: string;\n  \n  constructor(rootDir?: string) {\n    this.rootDir = rootDir || process.cwd();\n    this.versionManager = new VersionManager();\n    this.updateChecker = new UpdateChecker(this.versionManager);\n    this.dependencyChecker = new DependencyChecker(this.versionManager);\n    this.backupManager = new BackupManager(this.rootDir);\n  }\n  \n  /**\n   * Check for available updates\n   */\n  async checkForUpdates(): Promise<{ text: string }> {\n    try {\n      const result = await this.updateChecker.checkForUpdates();\n      const text = this.updateChecker.formatUpdateCheckResult(result);\n      return { text };\n    } catch (error) {\n      const text = this.updateChecker.formatUpdateCheckResult(null, error as Error);\n      return { text };\n    }\n  }\n  \n  /**\n   * Perform server update\n   */\n  async updateServer(createBackup: boolean = true, personaIndicator: string = ''): Promise<{ text: string }> {\n    const progress: UpdateProgress[] = [];\n    \n    try {\n      // Detect installation type\n      const installationType = InstallationDetector.getInstallationType();\n      logger.info(`[UpdateManager] Detected installation type: ${installationType}`);\n      \n      // Handle npm installations differently\n      if (installationType === 'npm') {\n        return this.updateNpmInstallation(createBackup, personaIndicator);\n      }\n      \n      // For git installations, proceed with existing logic\n      // Step 1: Check dependencies\n      progress.push({ step: 'dependencies', message: 'Checking system dependencies...', isComplete: false });\n      const dependencies = await this.dependencyChecker.checkDependencies();\n      \n      if (!dependencies.git.installed || dependencies.git.error) {\n        return {\n          text: personaIndicator + '❌ **Update Failed**\\n\\n' +\n            'Git is required for updates but is not available.\\n' +\n            dependencies.git.error || 'Git is not installed.'\n        };\n      }\n      \n      if (!dependencies.npm.installed || dependencies.npm.error) {\n        return {\n          text: personaIndicator + '❌ **Update Failed**\\n\\n' +\n            'npm is required for updates but is not available.\\n' +\n            dependencies.npm.error || 'npm is not installed.'\n        };\n      }\n      \n      progress[0].isComplete = true;\n      \n      // Step 2: Create backup if requested\n      if (createBackup) {\n        progress.push({ step: 'backup', message: 'Creating backup...', isComplete: false });\n        \n        const currentVersion = await this.versionManager.getCurrentVersion();\n        const backup = await this.backupManager.createBackup(currentVersion);\n        \n        progress[1].isComplete = true;\n        progress[1].message = `Backup created at: ${backup.timestamp}`;\n      }\n      \n      // Step 3: Git fetch\n      progress.push({ step: 'fetch', message: 'Fetching latest changes...', isComplete: false });\n      await safeExec('git', ['fetch', 'origin'], { cwd: this.rootDir });\n      progress[progress.length - 1].isComplete = true;\n      \n      // Step 4: Check for uncommitted changes\n      progress.push({ step: 'check', message: 'Checking for uncommitted changes...', isComplete: false });\n      const { stdout: statusOutput } = await safeExec('git', ['status', '--porcelain'], { cwd: this.rootDir });\n      \n      if (statusOutput.trim()) {\n        return {\n          text: personaIndicator + '❌ **Update Failed**\\n\\n' +\n            'You have uncommitted changes. Please commit or stash them before updating.\\n\\n' +\n            'Modified files:\\n' + statusOutput\n        };\n      }\n      progress[progress.length - 1].isComplete = true;\n      \n      // Step 5: Git pull\n      progress.push({ step: 'pull', message: 'Pulling latest changes...', isComplete: false });\n      const { stdout: pullOutput } = await safeExec('git', ['pull', 'origin', 'main'], { cwd: this.rootDir });\n      progress[progress.length - 1].isComplete = true;\n      \n      // Check if already up to date\n      if (pullOutput.includes('Already up to date')) {\n        return {\n          text: personaIndicator + '✅ **Already Up to Date**\\n\\n' +\n            'Your DollhouseMCP installation is already at the latest version.\\n\\n' +\n            'No changes were pulled from the repository.'\n        };\n      }\n      \n      // Step 6: npm install\n      progress.push({ step: 'install', message: 'Installing dependencies...', isComplete: false });\n      await safeExec('npm', ['install'], { cwd: this.rootDir });\n      progress[progress.length - 1].isComplete = true;\n      \n      // Step 7: Build\n      progress.push({ step: 'build', message: 'Building TypeScript...', isComplete: false });\n      await safeExec('npm', ['run', 'build'], { cwd: this.rootDir });\n      progress[progress.length - 1].isComplete = true;\n      \n      // Step 8: Cleanup old backups\n      if (createBackup) {\n        progress.push({ step: 'cleanup', message: 'Cleaning up old backups...', isComplete: false });\n        const deletedCount = await this.backupManager.cleanupOldBackups();\n        progress[progress.length - 1].isComplete = true;\n        progress[progress.length - 1].message = `Cleaned up ${deletedCount} old backup(s)`;\n      }\n      \n      // Format success message\n      const successParts = [\n        personaIndicator + '✅ **Update Complete!**\\n\\n',\n        '**Update Summary:**\\n'\n      ];\n      \n      progress.forEach(p => {\n        successParts.push(`${p.isComplete ? '✅' : '❌'} ${p.message}\\n`);\n      });\n      \n      successParts.push(\n        '\\n**Next Steps:**\\n',\n        '1. The server will restart automatically\\n',\n        '2. All personas will be reloaded\\n',\n        '3. Check `get_server_status` to verify the new version\\n\\n',\n        '💡 **Tip:** If you encounter issues, use `rollback_update true` to restore the previous version.'\n      );\n      \n      return { text: successParts.join('') };\n      \n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      \n      return {\n        text: personaIndicator + '❌ **Update Failed**\\n\\n' +\n          'Error: ' + errorMessage + '\\n\\n' +\n          '**Progress:**\\n' + \n          progress.map(p => `${p.isComplete ? '✅' : '❌'} ${p.message}`).join('\\n') + '\\n\\n' +\n          '**Recovery Options:**\\n' +\n          '• Try running the update again\\n' +\n          '• Check your internet connection\\n' +\n          '• Ensure you have proper permissions\\n' +\n          '• If a backup was created, use `rollback_update true` to restore'\n      };\n    }\n  }\n  \n  /**\n   * Rollback to previous version\n   */\n  async rollbackUpdate(force: boolean = false, personaIndicator: string = ''): Promise<{ text: string }> {\n    try {\n      // Check installation type\n      const installationType = InstallationDetector.getInstallationType();\n      \n      if (installationType === 'npm') {\n        return this.rollbackNpmInstallation(force, personaIndicator);\n      }\n      \n      // For git installations, use existing logic\n      // Get latest backup\n      const latestBackup = await this.backupManager.getLatestBackup();\n      \n      if (!latestBackup) {\n        return {\n          text: personaIndicator + '❌ **No Backups Found**\\n\\n' +\n            'There are no backups available to restore.\\n\\n' +\n            'Backups are created automatically when you run `update_server true`.'\n        };\n      }\n      \n      // Check if rollback is needed\n      if (!force) {\n        try {\n          // Test if the server is working by checking version\n          await this.versionManager.getCurrentVersion();\n          \n          return {\n            text: personaIndicator + '⚠️ **Rollback Confirmation Required**\\n\\n' +\n              'The server appears to be working normally.\\n\\n' +\n              `**Latest Backup:** ${latestBackup.timestamp}\\n` +\n              `**Backup Version:** ${latestBackup.version || 'Unknown'}\\n\\n` +\n              'To force rollback anyway, use: `rollback_update true`\\n\\n' +\n              '⚠️ **Warning:** This will restore all files to the backup state.'\n          };\n        } catch {\n          // Server is broken, proceed with rollback\n        }\n      }\n      \n      // Perform rollback\n      await this.backupManager.restoreBackup(latestBackup.path);\n      \n      // Reinstall dependencies\n      await safeExec('npm', ['install'], { cwd: this.rootDir });\n      \n      // Rebuild\n      await safeExec('npm', ['run', 'build'], { cwd: this.rootDir });\n      \n      return {\n        text: personaIndicator + '✅ **Rollback Complete!**\\n\\n' +\n          `Restored from backup: ${latestBackup.timestamp}\\n` +\n          `Backup version: ${latestBackup.version || 'Unknown'}\\n\\n` +\n          '**What was restored:**\\n' +\n          '• All source files\\n' +\n          '• Configuration files\\n' +\n          '• Dependencies reinstalled\\n' +\n          '• TypeScript rebuilt\\n\\n' +\n          '**Next Steps:**\\n' +\n          '1. The server will restart automatically\\n' +\n          '2. Check `get_server_status` to verify the version\\n' +\n          '3. Test your personas to ensure everything works'\n      };\n      \n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      \n      return {\n        text: personaIndicator + '❌ **Rollback Failed**\\n\\n' +\n          'Error: ' + errorMessage + '\\n\\n' +\n          '**Manual Recovery:**\\n' +\n          '1. Check the backups directory: ../dollhousemcp-backups/\\n' +\n          '2. Manually restore files if needed\\n' +\n          '3. Run `npm install` and `npm run build`\\n' +\n          '4. Contact support if issues persist'\n      };\n    }\n  }\n  \n  /**\n   * Update npm installation\n   */\n  private async updateNpmInstallation(createBackup: boolean, personaIndicator: string = ''): Promise<{ text: string }> {\n    try {\n      logger.info('[UpdateManager] Starting npm update process');\n      \n      // Check npm is available\n      const dependencies = await this.dependencyChecker.checkDependencies();\n      if (!dependencies.npm.installed || dependencies.npm.error) {\n        return {\n          text: personaIndicator + '❌ **Update Failed**\\n\\n' +\n            'npm is required for updates but is not available.\\n' +\n            dependencies.npm.error || 'npm is not installed.'\n        };\n      }\n      \n      // Get current version\n      const currentVersion = await this.versionManager.getCurrentVersion();\n      logger.info(`[UpdateManager] Current version: ${currentVersion}`);\n      \n      // Check latest version from npm registry\n      logger.info('[UpdateManager] Checking npm registry for latest version');\n      \n      // Security: Validate package name to prevent any potential injection\n      const packageName = '@dollhousemcp/mcp-server';\n      if (!/^@[a-z0-9-]+\\/[a-z0-9-]+$/.test(packageName)) {\n        throw new Error('Invalid package name format');\n      }\n      \n      const { stdout: npmViewOutput } = await safeExec('npm', ['view', packageName, 'version'], {\n        cwd: this.rootDir,\n        timeout: 30000\n      });\n      \n      const latestVersion = npmViewOutput.trim();\n      logger.info(`[UpdateManager] Latest npm version: ${latestVersion}`);\n      \n      // Compare versions\n      const comparison = compareVersions(currentVersion, latestVersion);\n      \n      if (comparison >= 0) {\n        return {\n          text: personaIndicator + '✅ **Already up to date!**\\n\\n' +\n            `Current version: ${currentVersion}\\n` +\n            `Latest version: ${latestVersion}\\n\\n` +\n            'No update needed.'\n        };\n      }\n      \n      // Get configuration\n      const config = UpdateConfigManager.getInstance();\n      \n      // Progress tracking\n      let progressMessage = personaIndicator + '🔄 **NPM Update in Progress**\\n\\n';\n      progressMessage += '**Steps:**\\n';\n      progressMessage += '✅ Version check complete\\n';\n      progressMessage += '⏳ Creating backup...\\n';\n      \n      // For npm installations, backup is mandatory for safety\n      logger.info('[UpdateManager] Creating backup before npm update');\n      try {\n        // For npm installations, we backup the global installation directory\n        const npmGlobalPath = InstallationDetector.getNpmGlobalPath();\n        if (!npmGlobalPath) {\n          throw new Error('Could not determine npm global installation path');\n        }\n        \n        // Create npm-specific backup with progress\n        const backupPath = await this.backupManager.createNpmBackup(npmGlobalPath, currentVersion);\n        logger.info(`[UpdateManager] Backup created at: ${backupPath}`);\n        \n        progressMessage = progressMessage.replace('⏳ Creating backup...', '✅ Backup created');\n        progressMessage += '⏳ Downloading and installing update...\\n';\n        \n      } catch (backupError) {\n        logger.error('[UpdateManager] Backup failed:', backupError);\n        return {\n          text: personaIndicator + '❌ **Update Failed**\\n\\n' +\n            'Failed to create backup before update.\\n' +\n            'Error: ' + (backupError instanceof Error ? backupError.message : String(backupError)) + '\\n\\n' +\n            '**Note:** Backup is mandatory for npm installations to ensure safe rollback.\\n' +\n            'Please check disk space and permissions.'\n        };\n      }\n      \n      // Perform npm update with progress\n      logger.info('[UpdateManager] Running npm update -g @dollhousemcp/mcp-server');\n      progressMessage += '\\n**Progress:**\\n';\n      progressMessage += '```\\n';\n      progressMessage += 'Running: npm update -g @dollhousemcp/mcp-server\\n';\n      progressMessage += 'This may take a few minutes...\\n';\n      progressMessage += '```\\n';\n      \n      const updateResult = await safeExec('npm', ['update', '-g', '@dollhousemcp/mcp-server'], {\n        cwd: this.rootDir,\n        timeout: config.getNpmUpdateTimeout()\n      });\n      \n      logger.info('[UpdateManager] npm update completed', updateResult);\n      \n      // Verify update succeeded\n      const { stdout: verifyOutput } = await safeExec('npm', ['list', '-g', '@dollhousemcp/mcp-server', '--depth=0'], {\n        cwd: this.rootDir,\n        timeout: 30000\n      });\n      \n      const versionMatch = verifyOutput.match(/@dollhousemcp\\/mcp-server@(\\d+\\.\\d+\\.\\d+)/);\n      const installedVersion = versionMatch ? versionMatch[1] : 'unknown';\n      \n      if (installedVersion !== latestVersion) {\n        logger.warn(`[UpdateManager] Version mismatch after update. Expected: ${latestVersion}, Got: ${installedVersion}`);\n      }\n      \n      return {\n        text: personaIndicator + '✅ **Update Complete!**\\n\\n' +\n          `Updated from v${currentVersion} to v${latestVersion}\\n\\n` +\n          '**What was updated:**\\n' +\n          '• DollhouseMCP server package\\n' +\n          '• All dependencies\\n\\n' +\n          '**Next Steps:**\\n' +\n          '1. The server will restart automatically\\n' +\n          '2. Check `get_server_status` to verify the new version\\n' +\n          '3. Test your personas to ensure everything works\\n\\n' +\n          '💡 **Tip:** If you encounter issues, use `rollback_update true` to restore the previous version.'\n      };\n      \n    } catch (error) {\n      logger.error('[UpdateManager] npm update failed:', error);\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      \n      return {\n        text: personaIndicator + '❌ **Update Failed**\\n\\n' +\n          'Error: ' + errorMessage + '\\n\\n' +\n          '**Troubleshooting:**\\n' +\n          '1. Ensure you have permission to update global npm packages\\n' +\n          '2. Try running with sudo if on macOS/Linux\\n' +\n          '3. Check your internet connection\\n' +\n          '4. Verify npm registry is accessible\\n\\n' +\n          '**Manual Update:**\\n' +\n          '```\\n' +\n          'npm update -g @dollhousemcp/mcp-server\\n' +\n          '```'\n      };\n    }\n  }\n  \n  /**\n   * Rollback npm installation\n   */\n  private async rollbackNpmInstallation(force: boolean, personaIndicator: string = ''): Promise<{ text: string }> {\n    try {\n      logger.info('[UpdateManager] Starting npm rollback process');\n      \n      // Get npm backup manifest from configuration\n      const config = UpdateConfigManager.getInstance();\n      const npmBackupsDir = config.getNpmBackupDir();\n      const manifestPath = path.join(npmBackupsDir, 'manifest.json');\n      \n      let manifest;\n      try {\n        const content = await fs.readFile(manifestPath, 'utf-8');\n        manifest = JSON.parse(content);\n      } catch (error) {\n        return {\n          text: personaIndicator + '❌ **No NPM Backups Found**\\n\\n' +\n            'There are no npm backups available to restore.\\n\\n' +\n            'Backups are created automatically when you run `update_server true` with npm installations.'\n        };\n      }\n      \n      if (!manifest.backups || manifest.backups.length === 0) {\n        return {\n          text: personaIndicator + '❌ **No NPM Backups Found**\\n\\n' +\n            'The backup manifest is empty.\\n\\n' +\n            'Backups are created automatically when you run `update_server true`.'\n        };\n      }\n      \n      // Get latest backup\n      const latestBackup = manifest.backups[0];\n      \n      // Check if rollback is needed\n      if (!force) {\n        try {\n          // Test if the server is working\n          await this.versionManager.getCurrentVersion();\n          \n          return {\n            text: personaIndicator + '⚠️ **Rollback Confirmation Required**\\n\\n' +\n              'The server appears to be working normally.\\n\\n' +\n              `**Latest Backup:** ${latestBackup.timestamp}\\n` +\n              `**Backup Version:** ${latestBackup.version || 'Unknown'}\\n\\n` +\n              'To force rollback anyway, use: `rollback_update true`\\n\\n' +\n              '⚠️ **Warning:** This will restore the npm package to the backup state.'\n          };\n        } catch {\n          // Server is broken, proceed with rollback\n        }\n      }\n      \n      // Get npm global path\n      const npmGlobalPath = InstallationDetector.getNpmGlobalPath();\n      if (!npmGlobalPath) {\n        return {\n          text: personaIndicator + '❌ **Rollback Failed**\\n\\n' +\n            'Could not determine npm global installation path.\\n\\n' +\n            'Please reinstall manually:\\n' +\n            '```\\n' +\n            'npm install -g @dollhousemcp/mcp-server@' + (latestBackup.version || 'latest') + '\\n' +\n            '```'\n        };\n      }\n      \n      logger.info(`[UpdateManager] Restoring npm backup from: ${latestBackup.path}`);\n      \n      // First validate that the backup is restorable\n      const backupPackagePath = path.join(latestBackup.path, 'package');\n      try {\n        await fs.access(backupPackagePath);\n        const packageJsonPath = path.join(backupPackagePath, 'package.json');\n        await fs.access(packageJsonPath);\n      } catch (error) {\n        return {\n          text: personaIndicator + '❌ **Backup Validation Failed**\\n\\n' +\n            'The backup appears to be corrupted or incomplete.\\n\\n' +\n            '**Details:**\\n' +\n            `Backup path: ${latestBackup.path}\\n` +\n            `Error: ${error instanceof Error ? error.message : String(error)}\\n\\n` +\n            '**Manual Recovery:**\\n' +\n            'Try another backup or reinstall manually:\\n' +\n            '```bash\\n' +\n            'npm install -g @dollhousemcp/mcp-server@' + (latestBackup.version || '1.4.0') + '\\n' +\n            '```'\n        };\n      }\n      \n      // Use transaction for atomic operations\n      const tempPath = `${npmGlobalPath}.tmp-${Date.now()}`;\n      const backupPath = `${npmGlobalPath}.backup-${Date.now()}`;\n      const transaction = FileOperations.createTransaction();\n      \n      try {\n        // Step 1: Copy backup to temporary location\n        await transaction.addCopy(backupPackagePath, tempPath);\n        \n        // Step 2: Move current installation to backup (atomic)\n        await transaction.addMove(npmGlobalPath, backupPath);\n        \n        // Step 3: Move temp to final location (atomic)\n        await transaction.addMove(tempPath, npmGlobalPath);\n        \n        // All operations successful, commit the transaction\n        transaction.commit();\n        \n        // Step 4: Clean up old backup (not part of transaction)\n        await fs.rm(backupPath, { recursive: true, force: true }).catch(() => {\n          logger.warn(`[UpdateManager] Failed to cleanup backup at: ${backupPath}`);\n        });\n      } catch (rollbackError) {\n        logger.error('[UpdateManager] Rollback operation failed, attempting recovery:', rollbackError);\n        \n        // Rollback all operations\n        if (transaction.hasOperations()) {\n          await transaction.rollback();\n        }\n        \n        // Additional recovery attempt\n        try {\n          // Check if npm path exists and is accessible\n          await fs.access(npmGlobalPath);\n          logger.info('[UpdateManager] NPM installation appears intact after rollback');\n        } catch {\n          // Try to restore from backup if main path is missing\n          try {\n            await fs.rename(backupPath, npmGlobalPath);\n            logger.info('[UpdateManager] Restored from backup path');\n          } catch {\n            logger.error('[UpdateManager] Complete rollback failure - manual intervention required');\n          }\n        }\n        \n        throw rollbackError;\n      }\n      \n      return {\n        text: personaIndicator + '✅ **NPM Rollback Complete!**\\n\\n' +\n          `Restored from backup: ${latestBackup.timestamp}\\n` +\n          `Backup version: ${latestBackup.version || 'Unknown'}\\n\\n` +\n          '**What was restored:**\\n' +\n          '• NPM package files\\n' +\n          '• All dependencies\\n\\n' +\n          '**Next Steps:**\\n' +\n          '1. The server will restart automatically\\n' +\n          '2. Check `get_server_status` to verify the version\\n' +\n          '3. Test your personas to ensure everything works'\n      };\n      \n    } catch (error) {\n      logger.error('[UpdateManager] npm rollback failed:', error);\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      \n      return {\n        text: personaIndicator + '❌ **NPM Rollback Failed**\\n\\n' +\n          'Error: ' + errorMessage + '\\n\\n' +\n          '**Manual Recovery:**\\n' +\n          '1. Check the backups directory: ~/.dollhouse/backups/npm/\\n' +\n          '2. Reinstall a specific version:\\n' +\n          '   ```\\n' +\n          '   npm install -g @dollhousemcp/mcp-server@1.4.0\\n' +\n          '   ```\\n' +\n          '3. Contact support if issues persist'\n      };\n    }\n  }\n  \n  /**\n   * Copy directory recursively with progress reporting\n   * @deprecated Use FileOperations.copyDirectory instead\n   */\n  private async copyDirectory(src: string, dest: string): Promise<void> {\n    await FileOperations.copyDirectory(src, dest, {\n      excludePatterns: ['.git', 'node_modules'],\n      onProgress: (copied, total, file) => {\n        logger.debug(`[UpdateManager] Copying files: ${copied}/${total} - ${path.basename(file)}`);\n      }\n    });\n  }\n  \n  /**\n   * Convert npm installation to git installation\n   */\n  async convertToGitInstallation(targetDir?: string, confirm: boolean = false, personaIndicator: string = ''): Promise<{ text: string }> {\n    try {\n      const installationType = InstallationDetector.getInstallationType();\n      \n      if (installationType === 'git') {\n        return {\n          text: personaIndicator + '⚠️ **Already a Git Installation**\\n\\n' +\n            'This server is already running from a git installation.\\n' +\n            'No conversion needed.'\n        };\n      }\n      \n      if (installationType === 'unknown') {\n        return {\n          text: personaIndicator + '❌ **Installation Type Unknown**\\n\\n' +\n            'Cannot determine the current installation type.\\n' +\n            'Please check your installation manually.'\n        };\n      }\n      \n      // Default target directory\n      const defaultTargetDir = path.join(process.env.HOME || '', '.dollhouse', 'mcp-server-git');\n      const gitTargetDir = targetDir || defaultTargetDir;\n      \n      if (!confirm) {\n        return {\n          text: personaIndicator + '🔄 **Convert to Git Installation**\\n\\n' +\n            '**This will:**\\n' +\n            `1. Clone DollhouseMCP to: ${gitTargetDir}\\n` +\n            '2. Copy your portfolio and settings\\n' +\n            '3. Build the TypeScript code\\n' +\n            '4. Provide Claude Desktop configuration\\n\\n' +\n            '**Benefits of Git Installation:**\\n' +\n            '• Full control over updates\\n' +\n            '• Access to development branches\\n' +\n            '• Ability to contribute changes\\n' +\n            '• Rollback to any commit\\n\\n' +\n            '**To proceed:**\\n' +\n            '`convert_to_git_installation true`\\n\\n' +\n            '**To use custom directory:**\\n' +\n            '`convert_to_git_installation \"/path/to/dir\" true`'\n        };\n      }\n      \n      logger.info(`[UpdateManager] Starting conversion to git installation at: ${gitTargetDir}`);\n      \n      // Check if target directory already exists\n      try {\n        await fs.access(gitTargetDir);\n        return {\n          text: personaIndicator + '❌ **Target Directory Exists**\\n\\n' +\n            `The directory ${gitTargetDir} already exists.\\n\\n` +\n            '**Options:**\\n' +\n            '1. Remove the existing directory first\\n' +\n            '2. Choose a different target directory\\n' +\n            '3. Use the existing git installation'\n        };\n      } catch {\n        // Directory doesn't exist, good to proceed\n      }\n      \n      // Progress message builder\n      let progressSteps = personaIndicator + '🔄 **Git Installation Progress**\\n\\n';\n      progressSteps += '**Steps:**\\n';\n      progressSteps += '⏳ Cloning repository...\\n';\n      progressSteps += '⏳ Installing dependencies...\\n';\n      progressSteps += '⏳ Building TypeScript...\\n';\n      progressSteps += '⏳ Setting up configuration...\\n\\n';\n      progressSteps += '**Current Step:** Cloning repository\\n';\n      progressSteps += '```\\n';\n      progressSteps += `Target: ${gitTargetDir}\\n`;\n      progressSteps += 'This may take a few minutes depending on your connection...\\n';\n      progressSteps += '```\\n';\n      \n      // Get configuration\n      const config = UpdateConfigManager.getInstance();\n      \n      // Step 1: Clone the repository\n      logger.info('[UpdateManager] Cloning repository...');\n      \n      // SECURITY FIX: Validate gitTargetDir to prevent command injection\n      // Previously: gitTargetDir passed directly to git clone command\n      // Now: Reject paths starting with '--' to prevent git option injection\n      if (gitTargetDir.startsWith('--')) {\n        throw new Error('Invalid target directory: cannot start with git options');\n      }\n      \n      await safeExec('git', ['clone', 'https://github.com/DollhouseMCP/mcp-server.git', gitTargetDir], {\n        timeout: config.getGitCloneTimeout()\n      });\n      \n      progressSteps = progressSteps.replace('⏳ Cloning repository...', '✅ Repository cloned');\n      progressSteps = progressSteps.replace('**Current Step:** Cloning repository', '**Current Step:** Installing dependencies');\n      \n      // Step 2: Install dependencies\n      logger.info('[UpdateManager] Installing dependencies...');\n      await safeExec('npm', ['install'], {\n        cwd: gitTargetDir,\n        timeout: config.getNpmInstallTimeout()\n      });\n      \n      progressSteps = progressSteps.replace('⏳ Installing dependencies...', '✅ Dependencies installed');\n      progressSteps = progressSteps.replace('**Current Step:** Installing dependencies', '**Current Step:** Building TypeScript');\n      \n      // Step 3: Build TypeScript\n      logger.info('[UpdateManager] Building TypeScript...');\n      await safeExec('npm', ['run', 'build'], {\n        cwd: gitTargetDir,\n        timeout: config.getBuildTimeout()\n      });\n      \n      progressSteps = progressSteps.replace('⏳ Building TypeScript...', '✅ TypeScript built');\n      progressSteps = progressSteps.replace('**Current Step:** Building TypeScript', '**Current Step:** Configuration');\n      \n      // Step 4: Copy portfolio\n      const portfolioSource = path.join(process.env.HOME || '', '.dollhouse', 'portfolio');\n      const portfolioTarget = path.join(process.env.HOME || '', '.dollhouse', 'portfolio');\n      \n      logger.info('[UpdateManager] Portfolio will remain at: ' + portfolioTarget);\n      \n      // Step 5: Generate Claude Desktop config\n      const configPath = path.join(gitTargetDir, 'dist', 'index.js');\n      const claudeConfig = {\n        mcpServers: {\n          dollhousemcp: {\n            command: 'node',\n            args: [configPath]\n          }\n        }\n      };\n      \n      return {\n        text: personaIndicator + '✅ **Git Installation Complete!**\\n\\n' +\n          `**Installation Location:** ${gitTargetDir}\\n\\n` +\n          '**Next Steps:**\\n\\n' +\n          '1. **Update Claude Desktop configuration:**\\n' +\n          '   ```json\\n' +\n          JSON.stringify(claudeConfig, null, 2) + '\\n' +\n          '   ```\\n\\n' +\n          '2. **Restart Claude Desktop**\\n\\n' +\n          '3. **Verify installation:**\\n' +\n          '   After restart, run `get_server_status` to confirm\\n\\n' +\n          '**Your portfolio remains at:**\\n' +\n          `   ${portfolioTarget}\\n\\n` +\n          '**To update in the future:**\\n' +\n          '   ```bash\\n' +\n          `   cd ${gitTargetDir}\\n` +\n          '   git pull\\n' +\n          '   npm install\\n' +\n          '   npm run build\\n' +\n          '   ```\\n\\n' +\n          '💡 **Tip:** You can now use `update_server` to update via git!'\n      };\n      \n    } catch (error) {\n      logger.error('[UpdateManager] Git conversion failed:', error);\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      \n      return {\n        text: personaIndicator + '❌ **Conversion Failed**\\n\\n' +\n          'Error: ' + errorMessage + '\\n\\n' +\n          '**Troubleshooting:**\\n' +\n          '1. Ensure git is installed\\n' +\n          '2. Check internet connection\\n' +\n          '3. Verify you have write permissions\\n' +\n          '4. Try a different target directory'\n      };\n    }\n  }\n  \n  /**\n   * Get current server status\n   */\n  async getServerStatus(personaIndicator: string = ''): Promise<{ text: string }> {\n    try {\n      const currentVersion = await this.versionManager.getCurrentVersion();\n      const dependencies = await this.dependencyChecker.checkDependencies();\n      const backups = await this.backupManager.listBackups();\n      const rateLimitStatus = this.updateChecker.getRateLimitStatus();\n      \n      // Get installation type\n      const installationType = InstallationDetector.getInstallationType();\n      const installationDesc = InstallationDetector.getInstallationDescription();\n      \n      // Get git status (only for git installations)\n      let gitStatus = 'N/A';\n      let gitBranch = 'N/A';\n      let lastCommit = 'N/A';\n      \n      if (installationType === 'git') {\n        try {\n          const { stdout: branchOutput } = await safeExec('git', ['branch', '--show-current'], { cwd: this.rootDir });\n          gitBranch = branchOutput.trim() || 'detached';\n          \n          const { stdout: statusOutput } = await safeExec('git', ['status', '--porcelain'], { cwd: this.rootDir });\n          gitStatus = statusOutput.trim() ? 'Modified' : 'Clean';\n          \n          const { stdout: logOutput } = await safeExec('git', ['log', '-1', '--oneline'], { cwd: this.rootDir });\n          lastCommit = logOutput.trim();\n        } catch {\n          // Git commands failed, use defaults\n        }\n      }\n      \n      const statusParts = [\n        personaIndicator + '📊 **DollhouseMCP Server Status**\\n\\n',\n        '**Version Information:**\\n',\n        `• Current Version: ${currentVersion}\\n`,\n        `• Installation Type: ${installationType} (${installationDesc})\\n`,\n        `• Git Branch: ${gitBranch}\\n`,\n        `• Git Status: ${gitStatus}\\n`,\n        `• Last Commit: ${lastCommit}\\n\\n`,\n        '**Dependencies:**\\n',\n        this.dependencyChecker.formatDependencyStatus(dependencies),\n        '\\n\\n**Backups:**\\n',\n        `• Total Backups: ${backups.length}\\n`\n      ];\n      \n      if (backups.length > 0) {\n        statusParts.push(`• Latest Backup: ${backups[0].timestamp} (v${backups[0].version || 'unknown'})\\n`);\n        statusParts.push(`• Oldest Backup: ${backups[backups.length - 1].timestamp}\\n`);\n      }\n      \n      statusParts.push(\n        '\\n**Rate Limit Status:**\\n',\n        `• Update Checks Remaining: ${rateLimitStatus.remainingRequests}/10 per hour\\n`,\n        `• Rate Limit Resets: ${rateLimitStatus.resetTime.toLocaleTimeString()}\\n`\n      );\n      \n      if (!rateLimitStatus.allowed && rateLimitStatus.waitTimeSeconds) {\n        statusParts.push(`• ⏳ Wait ${rateLimitStatus.waitTimeSeconds} seconds before next check\\n`);\n      }\n      \n      statusParts.push(\n        '\\n**Available Commands:**\\n',\n        '• `check_for_updates` - Check for new versions\\n',\n        '• `update_server true` - Update to latest version\\n',\n        '• `rollback_update true` - Restore from backup\\n'\n      );\n      \n      return { text: statusParts.join('') };\n      \n    } catch (error) {\n      const errorMessage = error instanceof Error ? error.message : String(error);\n      \n      return {\n        text: personaIndicator + '❌ **Status Check Failed**\\n\\n' +\n          'Error: ' + errorMessage + '\\n\\n' +\n          'The server may be in an inconsistent state.\\n' +\n          'Try running `update_server true` to fix issues.'\n      };\n    }\n  }\n}"]}