@dollhousemcp/mcp-server 1.4.1 → 1.4.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +54 -9
- package/dist/config/updateConfig.d.ts +84 -0
- package/dist/config/updateConfig.d.ts.map +1 -0
- package/dist/config/updateConfig.js +148 -0
- package/dist/generated/version.d.ts +9 -0
- package/dist/generated/version.d.ts.map +1 -0
- package/dist/generated/version.js +9 -0
- package/dist/index.d.ts +6 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +34 -6
- package/dist/portfolio/DefaultElementProvider.d.ts +78 -0
- package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -0
- package/dist/portfolio/DefaultElementProvider.js +398 -0
- package/dist/portfolio/PortfolioManager.d.ts +7 -0
- package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioManager.js +44 -3
- package/dist/security/commandValidator.d.ts.map +1 -1
- package/dist/security/commandValidator.js +5 -2
- package/dist/security/securityMonitor.d.ts +2 -1
- package/dist/security/securityMonitor.d.ts.map +1 -1
- package/dist/security/securityMonitor.js +1 -1
- package/dist/server/tools/UpdateTools.d.ts.map +1 -1
- package/dist/server/tools/UpdateTools.js +22 -1
- package/dist/server/types.d.ts +1 -0
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/types.js +1 -1
- package/dist/update/BackupManager.d.ts +17 -0
- package/dist/update/BackupManager.d.ts.map +1 -1
- package/dist/update/BackupManager.js +112 -3
- package/dist/update/UpdateManager.d.ts +19 -0
- package/dist/update/UpdateManager.d.ts.map +1 -1
- package/dist/update/UpdateManager.js +485 -15
- package/dist/update/VersionManager.d.ts +1 -1
- package/dist/update/VersionManager.d.ts.map +1 -1
- package/dist/update/VersionManager.js +62 -15
- package/dist/utils/fileOperations.d.ts +83 -0
- package/dist/utils/fileOperations.d.ts.map +1 -0
- package/dist/utils/fileOperations.js +291 -0
- package/dist/utils/installation.d.ts +30 -0
- package/dist/utils/installation.d.ts.map +1 -0
- package/dist/utils/installation.js +160 -0
- package/package.json +3 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"VersionManager.d.ts","sourceRoot":"","sources":["../../src/update/VersionManager.ts"],"names":[],"mappings":"AAAA;;GAEG;
|
|
1
|
+
{"version":3,"file":"VersionManager.d.ts","sourceRoot":"","sources":["../../src/update/VersionManager.ts"],"names":[],"mappings":"AAAA;;GAEG;AAQH,qBAAa,cAAc;IACzB;;OAEG;IACG,iBAAiB,IAAI,OAAO,CAAC,MAAM,CAAC;IA4E1C;;;OAGG;IACH,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,MAAM;IAgC3D;;OAEG;IACH,sBAAsB,CAAC,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IA2BnE;;OAEG;IACH,yBAAyB,CACvB,aAAa,EAAE,MAAM,EACrB,YAAY,EAAE;QAAE,OAAO,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAC;QAAC,WAAW,EAAE,MAAM,CAAA;KAAE,EACvE,QAAQ,EAAE,MAAM,GACf;QAAE,KAAK,EAAE,OAAO,CAAC;QAAC,OAAO,CAAC,EAAE,MAAM,CAAC;QAAC,KAAK,CAAC,EAAE,MAAM,CAAA;KAAE;CAgCxD"}
|
|
@@ -3,38 +3,85 @@
|
|
|
3
3
|
*/
|
|
4
4
|
import * as fs from 'fs/promises';
|
|
5
5
|
import * as path from 'path';
|
|
6
|
+
import { fileURLToPath } from 'url';
|
|
7
|
+
import { logger } from '../utils/logger.js';
|
|
8
|
+
import { InstallationDetector } from '../utils/installation.js';
|
|
6
9
|
export class VersionManager {
|
|
7
10
|
/**
|
|
8
|
-
* Get current version from package.json
|
|
11
|
+
* Get current version from package.json or embedded version
|
|
9
12
|
*/
|
|
10
13
|
async getCurrentVersion() {
|
|
11
|
-
//
|
|
14
|
+
// First, try to import the generated version file
|
|
15
|
+
try {
|
|
16
|
+
const { PACKAGE_VERSION } = await import('../generated/version.js');
|
|
17
|
+
if (PACKAGE_VERSION) {
|
|
18
|
+
logger.debug(`[VersionManager] Using embedded version: ${PACKAGE_VERSION}`);
|
|
19
|
+
return PACKAGE_VERSION;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
catch (error) {
|
|
23
|
+
logger.debug('[VersionManager] No embedded version found, will search for package.json');
|
|
24
|
+
}
|
|
25
|
+
// Determine installation type
|
|
26
|
+
const installationType = InstallationDetector.getInstallationType();
|
|
27
|
+
// For npm installations, look relative to the module location
|
|
28
|
+
if (installationType === 'npm') {
|
|
29
|
+
const npmPath = InstallationDetector.getNpmGlobalPath();
|
|
30
|
+
if (npmPath) {
|
|
31
|
+
const packageJsonPath = path.join(npmPath, 'package.json');
|
|
32
|
+
try {
|
|
33
|
+
const packageContent = await fs.readFile(packageJsonPath, 'utf-8');
|
|
34
|
+
const packageData = JSON.parse(packageContent);
|
|
35
|
+
return packageData.version;
|
|
36
|
+
}
|
|
37
|
+
catch (error) {
|
|
38
|
+
logger.error('[VersionManager] Error reading package.json from npm path:', error);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
// For git installations, search from current file location
|
|
43
|
+
if (installationType === 'git') {
|
|
44
|
+
const currentFileUrl = import.meta.url;
|
|
45
|
+
const currentFilePath = fileURLToPath(currentFileUrl);
|
|
46
|
+
let currentDir = path.dirname(currentFilePath);
|
|
47
|
+
// Search upward for package.json
|
|
48
|
+
for (let i = 0; i < 5; i++) {
|
|
49
|
+
const candidatePath = path.join(currentDir, 'package.json');
|
|
50
|
+
try {
|
|
51
|
+
await fs.access(candidatePath);
|
|
52
|
+
const packageContent = await fs.readFile(candidatePath, 'utf-8');
|
|
53
|
+
const packageData = JSON.parse(packageContent);
|
|
54
|
+
return packageData.version;
|
|
55
|
+
}
|
|
56
|
+
catch {
|
|
57
|
+
// File doesn't exist, try parent directory
|
|
58
|
+
const parentDir = path.dirname(currentDir);
|
|
59
|
+
if (parentDir === currentDir) {
|
|
60
|
+
break;
|
|
61
|
+
}
|
|
62
|
+
currentDir = parentDir;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
// Last resort: try process.cwd() (original behavior)
|
|
12
67
|
let currentDir = process.cwd();
|
|
13
|
-
let packageJsonPath = null;
|
|
14
|
-
// Search up to 5 levels for package.json
|
|
15
68
|
for (let i = 0; i < 5; i++) {
|
|
16
69
|
const candidatePath = path.join(currentDir, 'package.json');
|
|
17
70
|
try {
|
|
18
71
|
await fs.access(candidatePath);
|
|
19
|
-
|
|
20
|
-
|
|
72
|
+
const packageContent = await fs.readFile(candidatePath, 'utf-8');
|
|
73
|
+
const packageData = JSON.parse(packageContent);
|
|
74
|
+
return packageData.version;
|
|
21
75
|
}
|
|
22
76
|
catch {
|
|
23
|
-
// File doesn't exist, try parent directory
|
|
24
77
|
const parentDir = path.dirname(currentDir);
|
|
25
78
|
if (parentDir === currentDir) {
|
|
26
|
-
// We've reached the root
|
|
27
79
|
break;
|
|
28
80
|
}
|
|
29
81
|
currentDir = parentDir;
|
|
30
82
|
}
|
|
31
83
|
}
|
|
32
|
-
|
|
33
|
-
throw new Error('Could not find package.json in current directory or any parent directory');
|
|
34
|
-
}
|
|
35
|
-
const packageContent = await fs.readFile(packageJsonPath, 'utf-8');
|
|
36
|
-
const packageData = JSON.parse(packageContent);
|
|
37
|
-
return packageData.version;
|
|
84
|
+
throw new Error('Could not determine version. Please ensure you have a valid installation.');
|
|
38
85
|
}
|
|
39
86
|
/**
|
|
40
87
|
* Enhanced semantic version comparison supporting pre-release versions
|
|
@@ -131,4 +178,4 @@ export class VersionManager {
|
|
|
131
178
|
return { valid: true };
|
|
132
179
|
}
|
|
133
180
|
}
|
|
134
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"VersionManager.js","sourceRoot":"","sources":["../../src/update/VersionManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Version management and comparison utilities\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\n\nexport class VersionManager {\n  /**\n   * Get current version from package.json\n   */\n  async getCurrentVersion(): Promise<string> {\n    // Use process.cwd() as a base, then search upward for package.json\n    let currentDir = process.cwd();\n    let packageJsonPath: string | null = null;\n    \n    // Search up to 5 levels for package.json\n    for (let i = 0; i < 5; i++) {\n      const candidatePath = path.join(currentDir, 'package.json');\n      try {\n        await fs.access(candidatePath);\n        packageJsonPath = candidatePath;\n        break;\n      } catch {\n        // File doesn't exist, try parent directory\n        const parentDir = path.dirname(currentDir);\n        if (parentDir === currentDir) {\n          // We've reached the root\n          break;\n        }\n        currentDir = parentDir;\n      }\n    }\n    \n    if (!packageJsonPath) {\n      throw new Error('Could not find package.json in current directory or any parent directory');\n    }\n    \n    const packageContent = await fs.readFile(packageJsonPath, 'utf-8');\n    const packageData = JSON.parse(packageContent);\n    return packageData.version;\n  }\n  \n  /**\n   * Enhanced semantic version comparison supporting pre-release versions\n   * Returns: -1 if v1 < v2, 0 if v1 == v2, 1 if v1 > v2\n   */\n  compareVersions(version1: string, version2: string): number {\n    // Normalize versions by removing 'v' prefix\n    const v1 = version1.replace(/^v/, '');\n    const v2 = version2.replace(/^v/, '');\n    \n    // Split version and pre-release parts\n    const [v1main, v1pre] = v1.split('-');\n    const [v2main, v2pre] = v2.split('-');\n    \n    // Compare main version parts (x.y.z)\n    const v1parts = v1main.split('.').map(part => parseInt(part) || 0);\n    const v2parts = v2main.split('.').map(part => parseInt(part) || 0);\n    \n    const maxLength = Math.max(v1parts.length, v2parts.length);\n    for (let i = 0; i < maxLength; i++) {\n      const v1part = v1parts[i] || 0;\n      const v2part = v2parts[i] || 0;\n      \n      if (v1part < v2part) return -1;\n      if (v1part > v2part) return 1;\n    }\n    \n    // If main versions are equal, compare pre-release versions\n    // Version without pre-release is greater than version with pre-release\n    if (!v1pre && v2pre) return 1;   // 1.0.0 > 1.0.0-beta\n    if (v1pre && !v2pre) return -1;  // 1.0.0-beta < 1.0.0\n    if (!v1pre && !v2pre) return 0;  // 1.0.0 == 1.0.0\n    \n    // Both have pre-release, compare lexicographically\n    return v1pre.localeCompare(v2pre);\n  }\n  \n  /**\n   * Parse version from dependency output\n   */\n  parseVersionFromOutput(output: string, tool: string): string | null {\n    // Git version output: \"git version 2.39.2\"\n    // npm version output: \"8.19.2\" or JSON with version info\n    \n    if (tool === 'git') {\n      const match = output.match(/git version (\\d+\\.\\d+\\.\\d+)/);\n      return match ? match[1] : null;\n    } else if (tool === 'npm') {\n      // npm might return just the version number or JSON\n      const cleanOutput = output.trim();\n      if (cleanOutput.match(/^\\d+\\.\\d+\\.\\d+/)) {\n        return cleanOutput.split('\\n')[0]; // First line if multiple lines\n      }\n      // Try to parse as JSON if it looks like JSON\n      try {\n        const parsed = JSON.parse(cleanOutput);\n        return parsed.npm || parsed.version || null;\n      } catch {\n        // If not JSON, try to extract version pattern\n        const match = cleanOutput.match(/(\\d+\\.\\d+\\.\\d+)/);\n        return match ? match[1] : null;\n      }\n    }\n    \n    return null;\n  }\n  \n  /**\n   * Validate that a dependency version meets requirements\n   */\n  validateDependencyVersion(\n    actualVersion: string, \n    requirements: { minimum: string; maximum: string; recommended: string },\n    toolName: string\n  ): { valid: boolean; warning?: string; error?: string } {\n    const minComparison = this.compareVersions(actualVersion, requirements.minimum);\n    const maxComparison = this.compareVersions(actualVersion, requirements.maximum);\n    \n    // Check if version is below minimum\n    if (minComparison < 0) {\n      return {\n        valid: false,\n        error: `${toolName} version ${actualVersion} is below minimum required version ${requirements.minimum}. Please upgrade ${toolName}.`\n      };\n    }\n    \n    // Check if version is above maximum tested\n    if (maxComparison > 0) {\n      return {\n        valid: true,\n        warning: `${toolName} version ${actualVersion} is above maximum tested version ${requirements.maximum}. Some features may not work as expected.`\n      };\n    }\n    \n    // Check if not at recommended version\n    const recComparison = this.compareVersions(actualVersion, requirements.recommended);\n    if (recComparison !== 0) {\n      return {\n        valid: true,\n        warning: `${toolName} version ${actualVersion} works but ${requirements.recommended} is recommended for optimal stability.`\n      };\n    }\n    \n    // Version is perfect\n    return { valid: true };\n  }\n}"]}
|
|
181
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"VersionManager.js","sourceRoot":"","sources":["../../src/update/VersionManager.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,EAAE,MAAM,aAAa,CAAC;AAClC,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAC7B,OAAO,EAAE,aAAa,EAAE,MAAM,KAAK,CAAC;AACpC,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAC5C,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAEhE,MAAM,OAAO,cAAc;IACzB;;OAEG;IACH,KAAK,CAAC,iBAAiB;QACrB,kDAAkD;QAClD,IAAI,CAAC;YACH,MAAM,EAAE,eAAe,EAAE,GAAG,MAAM,MAAM,CAAC,yBAAyB,CAAC,CAAC;YACpE,IAAI,eAAe,EAAE,CAAC;gBACpB,MAAM,CAAC,KAAK,CAAC,4CAA4C,eAAe,EAAE,CAAC,CAAC;gBAC5E,OAAO,eAAe,CAAC;YACzB,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,KAAK,CAAC,0EAA0E,CAAC,CAAC;QAC3F,CAAC;QAED,8BAA8B;QAC9B,MAAM,gBAAgB,GAAG,oBAAoB,CAAC,mBAAmB,EAAE,CAAC;QAEpE,8DAA8D;QAC9D,IAAI,gBAAgB,KAAK,KAAK,EAAE,CAAC;YAC/B,MAAM,OAAO,GAAG,oBAAoB,CAAC,gBAAgB,EAAE,CAAC;YACxD,IAAI,OAAO,EAAE,CAAC;gBACZ,MAAM,eAAe,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC;gBAC3D,IAAI,CAAC;oBACH,MAAM,cAAc,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;oBACnE,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;oBAC/C,OAAO,WAAW,CAAC,OAAO,CAAC;gBAC7B,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACf,MAAM,CAAC,KAAK,CAAC,4DAA4D,EAAE,KAAK,CAAC,CAAC;gBACpF,CAAC;YACH,CAAC;QACH,CAAC;QAED,2DAA2D;QAC3D,IAAI,gBAAgB,KAAK,KAAK,EAAE,CAAC;YAC/B,MAAM,cAAc,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC;YACvC,MAAM,eAAe,GAAG,aAAa,CAAC,cAAc,CAAC,CAAC;YACtD,IAAI,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC;YAE/C,iCAAiC;YACjC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;gBAC3B,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,cAAc,CAAC,CAAC;gBAC5D,IAAI,CAAC;oBACH,MAAM,EAAE,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;oBAC/B,MAAM,cAAc,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,aAAa,EAAE,OAAO,CAAC,CAAC;oBACjE,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;oBAC/C,OAAO,WAAW,CAAC,OAAO,CAAC;gBAC7B,CAAC;gBAAC,MAAM,CAAC;oBACP,2CAA2C;oBAC3C,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;oBAC3C,IAAI,SAAS,KAAK,UAAU,EAAE,CAAC;wBAC7B,MAAM;oBACR,CAAC;oBACD,UAAU,GAAG,SAAS,CAAC;gBACzB,CAAC;YACH,CAAC;QACH,CAAC;QAED,qDAAqD;QACrD,IAAI,UAAU,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;QAC/B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;YAC3B,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,cAAc,CAAC,CAAC;YAC5D,IAAI,CAAC;gBACH,MAAM,EAAE,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;gBAC/B,MAAM,cAAc,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC,aAAa,EAAE,OAAO,CAAC,CAAC;gBACjE,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;gBAC/C,OAAO,WAAW,CAAC,OAAO,CAAC;YAC7B,CAAC;YAAC,MAAM,CAAC;gBACP,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;gBAC3C,IAAI,SAAS,KAAK,UAAU,EAAE,CAAC;oBAC7B,MAAM;gBACR,CAAC;gBACD,UAAU,GAAG,SAAS,CAAC;YACzB,CAAC;QACH,CAAC;QAED,MAAM,IAAI,KAAK,CAAC,2EAA2E,CAAC,CAAC;IAC/F,CAAC;IAED;;;OAGG;IACH,eAAe,CAAC,QAAgB,EAAE,QAAgB;QAChD,4CAA4C;QAC5C,MAAM,EAAE,GAAG,QAAQ,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QACtC,MAAM,EAAE,GAAG,QAAQ,CAAC,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;QAEtC,sCAAsC;QACtC,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACtC,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QAEtC,qCAAqC;QACrC,MAAM,OAAO,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;QACnE,MAAM,OAAO,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;QAEnE,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;QAC3D,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,SAAS,EAAE,CAAC,EAAE,EAAE,CAAC;YACnC,MAAM,MAAM,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YAC/B,MAAM,MAAM,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YAE/B,IAAI,MAAM,GAAG,MAAM;gBAAE,OAAO,CAAC,CAAC,CAAC;YAC/B,IAAI,MAAM,GAAG,MAAM;gBAAE,OAAO,CAAC,CAAC;QAChC,CAAC;QAED,2DAA2D;QAC3D,uEAAuE;QACvE,IAAI,CAAC,KAAK,IAAI,KAAK;YAAE,OAAO,CAAC,CAAC,CAAG,qBAAqB;QACtD,IAAI,KAAK,IAAI,CAAC,KAAK;YAAE,OAAO,CAAC,CAAC,CAAC,CAAE,qBAAqB;QACtD,IAAI,CAAC,KAAK,IAAI,CAAC,KAAK;YAAE,OAAO,CAAC,CAAC,CAAE,iBAAiB;QAElD,mDAAmD;QACnD,OAAO,KAAK,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;IACpC,CAAC;IAED;;OAEG;IACH,sBAAsB,CAAC,MAAc,EAAE,IAAY;QACjD,2CAA2C;QAC3C,yDAAyD;QAEzD,IAAI,IAAI,KAAK,KAAK,EAAE,CAAC;YACnB,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,6BAA6B,CAAC,CAAC;YAC1D,OAAO,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QACjC,CAAC;aAAM,IAAI,IAAI,KAAK,KAAK,EAAE,CAAC;YAC1B,mDAAmD;YACnD,MAAM,WAAW,GAAG,MAAM,CAAC,IAAI,EAAE,CAAC;YAClC,IAAI,WAAW,CAAC,KAAK,CAAC,gBAAgB,CAAC,EAAE,CAAC;gBACxC,OAAO,WAAW,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,+BAA+B;YACpE,CAAC;YACD,6CAA6C;YAC7C,IAAI,CAAC;gBACH,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;gBACvC,OAAO,MAAM,CAAC,GAAG,IAAI,MAAM,CAAC,OAAO,IAAI,IAAI,CAAC;YAC9C,CAAC;YAAC,MAAM,CAAC;gBACP,8CAA8C;gBAC9C,MAAM,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAC;gBACnD,OAAO,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;YACjC,CAAC;QACH,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,yBAAyB,CACvB,aAAqB,EACrB,YAAuE,EACvE,QAAgB;QAEhB,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,YAAY,CAAC,OAAO,CAAC,CAAC;QAChF,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,YAAY,CAAC,OAAO,CAAC,CAAC;QAEhF,oCAAoC;QACpC,IAAI,aAAa,GAAG,CAAC,EAAE,CAAC;YACtB,OAAO;gBACL,KAAK,EAAE,KAAK;gBACZ,KAAK,EAAE,GAAG,QAAQ,YAAY,aAAa,sCAAsC,YAAY,CAAC,OAAO,oBAAoB,QAAQ,GAAG;aACrI,CAAC;QACJ,CAAC;QAED,2CAA2C;QAC3C,IAAI,aAAa,GAAG,CAAC,EAAE,CAAC;YACtB,OAAO;gBACL,KAAK,EAAE,IAAI;gBACX,OAAO,EAAE,GAAG,QAAQ,YAAY,aAAa,oCAAoC,YAAY,CAAC,OAAO,2CAA2C;aACjJ,CAAC;QACJ,CAAC;QAED,sCAAsC;QACtC,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,YAAY,CAAC,WAAW,CAAC,CAAC;QACpF,IAAI,aAAa,KAAK,CAAC,EAAE,CAAC;YACxB,OAAO;gBACL,KAAK,EAAE,IAAI;gBACX,OAAO,EAAE,GAAG,QAAQ,YAAY,aAAa,cAAc,YAAY,CAAC,WAAW,wCAAwC;aAC5H,CAAC;QACJ,CAAC;QAED,qBAAqB;QACrB,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC;IACzB,CAAC;CACF","sourcesContent":["/**\n * Version management and comparison utilities\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { fileURLToPath } from 'url';\nimport { logger } from '../utils/logger.js';\nimport { InstallationDetector } from '../utils/installation.js';\n\nexport class VersionManager {\n  /**\n   * Get current version from package.json or embedded version\n   */\n  async getCurrentVersion(): Promise<string> {\n    // First, try to import the generated version file\n    try {\n      const { PACKAGE_VERSION } = await import('../generated/version.js');\n      if (PACKAGE_VERSION) {\n        logger.debug(`[VersionManager] Using embedded version: ${PACKAGE_VERSION}`);\n        return PACKAGE_VERSION;\n      }\n    } catch (error) {\n      logger.debug('[VersionManager] No embedded version found, will search for package.json');\n    }\n    \n    // Determine installation type\n    const installationType = InstallationDetector.getInstallationType();\n    \n    // For npm installations, look relative to the module location\n    if (installationType === 'npm') {\n      const npmPath = InstallationDetector.getNpmGlobalPath();\n      if (npmPath) {\n        const packageJsonPath = path.join(npmPath, 'package.json');\n        try {\n          const packageContent = await fs.readFile(packageJsonPath, 'utf-8');\n          const packageData = JSON.parse(packageContent);\n          return packageData.version;\n        } catch (error) {\n          logger.error('[VersionManager] Error reading package.json from npm path:', error);\n        }\n      }\n    }\n    \n    // For git installations, search from current file location\n    if (installationType === 'git') {\n      const currentFileUrl = import.meta.url;\n      const currentFilePath = fileURLToPath(currentFileUrl);\n      let currentDir = path.dirname(currentFilePath);\n      \n      // Search upward for package.json\n      for (let i = 0; i < 5; i++) {\n        const candidatePath = path.join(currentDir, 'package.json');\n        try {\n          await fs.access(candidatePath);\n          const packageContent = await fs.readFile(candidatePath, 'utf-8');\n          const packageData = JSON.parse(packageContent);\n          return packageData.version;\n        } catch {\n          // File doesn't exist, try parent directory\n          const parentDir = path.dirname(currentDir);\n          if (parentDir === currentDir) {\n            break;\n          }\n          currentDir = parentDir;\n        }\n      }\n    }\n    \n    // Last resort: try process.cwd() (original behavior)\n    let currentDir = process.cwd();\n    for (let i = 0; i < 5; i++) {\n      const candidatePath = path.join(currentDir, 'package.json');\n      try {\n        await fs.access(candidatePath);\n        const packageContent = await fs.readFile(candidatePath, 'utf-8');\n        const packageData = JSON.parse(packageContent);\n        return packageData.version;\n      } catch {\n        const parentDir = path.dirname(currentDir);\n        if (parentDir === currentDir) {\n          break;\n        }\n        currentDir = parentDir;\n      }\n    }\n    \n    throw new Error('Could not determine version. Please ensure you have a valid installation.');\n  }\n  \n  /**\n   * Enhanced semantic version comparison supporting pre-release versions\n   * Returns: -1 if v1 < v2, 0 if v1 == v2, 1 if v1 > v2\n   */\n  compareVersions(version1: string, version2: string): number {\n    // Normalize versions by removing 'v' prefix\n    const v1 = version1.replace(/^v/, '');\n    const v2 = version2.replace(/^v/, '');\n    \n    // Split version and pre-release parts\n    const [v1main, v1pre] = v1.split('-');\n    const [v2main, v2pre] = v2.split('-');\n    \n    // Compare main version parts (x.y.z)\n    const v1parts = v1main.split('.').map(part => parseInt(part) || 0);\n    const v2parts = v2main.split('.').map(part => parseInt(part) || 0);\n    \n    const maxLength = Math.max(v1parts.length, v2parts.length);\n    for (let i = 0; i < maxLength; i++) {\n      const v1part = v1parts[i] || 0;\n      const v2part = v2parts[i] || 0;\n      \n      if (v1part < v2part) return -1;\n      if (v1part > v2part) return 1;\n    }\n    \n    // If main versions are equal, compare pre-release versions\n    // Version without pre-release is greater than version with pre-release\n    if (!v1pre && v2pre) return 1;   // 1.0.0 > 1.0.0-beta\n    if (v1pre && !v2pre) return -1;  // 1.0.0-beta < 1.0.0\n    if (!v1pre && !v2pre) return 0;  // 1.0.0 == 1.0.0\n    \n    // Both have pre-release, compare lexicographically\n    return v1pre.localeCompare(v2pre);\n  }\n  \n  /**\n   * Parse version from dependency output\n   */\n  parseVersionFromOutput(output: string, tool: string): string | null {\n    // Git version output: \"git version 2.39.2\"\n    // npm version output: \"8.19.2\" or JSON with version info\n    \n    if (tool === 'git') {\n      const match = output.match(/git version (\\d+\\.\\d+\\.\\d+)/);\n      return match ? match[1] : null;\n    } else if (tool === 'npm') {\n      // npm might return just the version number or JSON\n      const cleanOutput = output.trim();\n      if (cleanOutput.match(/^\\d+\\.\\d+\\.\\d+/)) {\n        return cleanOutput.split('\\n')[0]; // First line if multiple lines\n      }\n      // Try to parse as JSON if it looks like JSON\n      try {\n        const parsed = JSON.parse(cleanOutput);\n        return parsed.npm || parsed.version || null;\n      } catch {\n        // If not JSON, try to extract version pattern\n        const match = cleanOutput.match(/(\\d+\\.\\d+\\.\\d+)/);\n        return match ? match[1] : null;\n      }\n    }\n    \n    return null;\n  }\n  \n  /**\n   * Validate that a dependency version meets requirements\n   */\n  validateDependencyVersion(\n    actualVersion: string, \n    requirements: { minimum: string; maximum: string; recommended: string },\n    toolName: string\n  ): { valid: boolean; warning?: string; error?: string } {\n    const minComparison = this.compareVersions(actualVersion, requirements.minimum);\n    const maxComparison = this.compareVersions(actualVersion, requirements.maximum);\n    \n    // Check if version is below minimum\n    if (minComparison < 0) {\n      return {\n        valid: false,\n        error: `${toolName} version ${actualVersion} is below minimum required version ${requirements.minimum}. Please upgrade ${toolName}.`\n      };\n    }\n    \n    // Check if version is above maximum tested\n    if (maxComparison > 0) {\n      return {\n        valid: true,\n        warning: `${toolName} version ${actualVersion} is above maximum tested version ${requirements.maximum}. Some features may not work as expected.`\n      };\n    }\n    \n    // Check if not at recommended version\n    const recComparison = this.compareVersions(actualVersion, requirements.recommended);\n    if (recComparison !== 0) {\n      return {\n        valid: true,\n        warning: `${toolName} version ${actualVersion} works but ${requirements.recommended} is recommended for optimal stability.`\n      };\n    }\n    \n    // Version is perfect\n    return { valid: true };\n  }\n}"]}
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
export interface CopyOptions {
|
|
2
|
+
onProgress?: (copied: number, total: number, currentFile: string) => void;
|
|
3
|
+
excludePatterns?: string[];
|
|
4
|
+
maxRetries?: number;
|
|
5
|
+
}
|
|
6
|
+
export interface FileStats {
|
|
7
|
+
totalFiles: number;
|
|
8
|
+
totalSize: number;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Cross-platform file operations utility
|
|
12
|
+
* Centralizes common file operations with progress reporting and error handling
|
|
13
|
+
*/
|
|
14
|
+
export declare class FileOperations {
|
|
15
|
+
/**
|
|
16
|
+
* Recursively copy a directory with progress reporting
|
|
17
|
+
* Works cross-platform without relying on shell commands
|
|
18
|
+
*/
|
|
19
|
+
static copyDirectory(src: string, dest: string, options?: CopyOptions): Promise<void>;
|
|
20
|
+
/**
|
|
21
|
+
* Calculate directory statistics for progress reporting
|
|
22
|
+
*/
|
|
23
|
+
private static calculateDirectoryStats;
|
|
24
|
+
/**
|
|
25
|
+
* Internal recursive copy implementation
|
|
26
|
+
*/
|
|
27
|
+
private static copyDirectoryRecursive;
|
|
28
|
+
/**
|
|
29
|
+
* Copy a single file with retry logic
|
|
30
|
+
*/
|
|
31
|
+
private static copyFileWithRetry;
|
|
32
|
+
/**
|
|
33
|
+
* Check if a file/directory should be excluded
|
|
34
|
+
*/
|
|
35
|
+
private static shouldExclude;
|
|
36
|
+
/**
|
|
37
|
+
* Remove a directory with progress reporting
|
|
38
|
+
*/
|
|
39
|
+
static removeDirectory(dir: string, options?: {
|
|
40
|
+
onProgress?: (removed: number, total: number) => void;
|
|
41
|
+
}): Promise<void>;
|
|
42
|
+
/**
|
|
43
|
+
* Internal recursive remove implementation
|
|
44
|
+
*/
|
|
45
|
+
private static removeDirectoryRecursive;
|
|
46
|
+
/**
|
|
47
|
+
* Create a transaction manager for atomic file operations
|
|
48
|
+
*/
|
|
49
|
+
static createTransaction(): FileTransaction;
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Transaction manager for atomic file operations
|
|
53
|
+
* Ensures all operations succeed or all are rolled back
|
|
54
|
+
*/
|
|
55
|
+
export declare class FileTransaction {
|
|
56
|
+
private operations;
|
|
57
|
+
private completed;
|
|
58
|
+
/**
|
|
59
|
+
* Add a move operation to the transaction
|
|
60
|
+
*/
|
|
61
|
+
addMove(source: string, destination: string): Promise<void>;
|
|
62
|
+
/**
|
|
63
|
+
* Add a copy operation to the transaction
|
|
64
|
+
*/
|
|
65
|
+
addCopy(source: string, destination: string): Promise<void>;
|
|
66
|
+
/**
|
|
67
|
+
* Add a delete operation to the transaction
|
|
68
|
+
*/
|
|
69
|
+
addDelete(path: string, backupPath?: string): Promise<void>;
|
|
70
|
+
/**
|
|
71
|
+
* Commit the transaction (mark as successful)
|
|
72
|
+
*/
|
|
73
|
+
commit(): void;
|
|
74
|
+
/**
|
|
75
|
+
* Rollback all operations in reverse order
|
|
76
|
+
*/
|
|
77
|
+
rollback(): Promise<void>;
|
|
78
|
+
/**
|
|
79
|
+
* Check if any operations have been performed
|
|
80
|
+
*/
|
|
81
|
+
hasOperations(): boolean;
|
|
82
|
+
}
|
|
83
|
+
//# sourceMappingURL=fileOperations.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fileOperations.d.ts","sourceRoot":"","sources":["../../src/utils/fileOperations.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,WAAW;IAC1B,UAAU,CAAC,EAAE,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,KAAK,IAAI,CAAC;IAC1E,eAAe,CAAC,EAAE,MAAM,EAAE,CAAC;IAC3B,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,SAAS;IACxB,UAAU,EAAE,MAAM,CAAC;IACnB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED;;;GAGG;AACH,qBAAa,cAAc;IACzB;;;OAGG;WACU,aAAa,CACxB,GAAG,EAAE,MAAM,EACX,IAAI,EAAE,MAAM,EACZ,OAAO,GAAE,WAAgB,GACxB,OAAO,CAAC,IAAI,CAAC;IAqBhB;;OAEG;mBACkB,uBAAuB;IAuC5C;;OAEG;mBACkB,sBAAsB;IAqC3C;;OAEG;mBACkB,iBAAiB;IA0BtC;;OAEG;IACH,OAAO,CAAC,MAAM,CAAC,aAAa;IAa5B;;OAEG;WACU,eAAe,CAC1B,GAAG,EAAE,MAAM,EACX,OAAO,GAAE;QAAE,UAAU,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,KAAK,IAAI,CAAA;KAAO,GACtE,OAAO,CAAC,IAAI,CAAC;IAYhB;;OAEG;mBACkB,wBAAwB;IA0B7C;;OAEG;IACH,MAAM,CAAC,iBAAiB,IAAI,eAAe;CAG5C;AAED;;;GAGG;AACH,qBAAa,eAAe;IAC1B,OAAO,CAAC,UAAU,CAKV;IAER,OAAO,CAAC,SAAS,CAAS;IAE1B;;OAEG;IACG,OAAO,CAAC,MAAM,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAuBjE;;OAEG;IACG,OAAO,CAAC,MAAM,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAuBjE;;OAEG;IACG,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAmCjE;;OAEG;IACH,MAAM,IAAI,IAAI;IAId;;OAEG;IACG,QAAQ,IAAI,OAAO,CAAC,IAAI,CAAC;IAmB/B;;OAEG;IACH,aAAa,IAAI,OAAO;CAGzB"}
|
|
@@ -0,0 +1,291 @@
|
|
|
1
|
+
import * as fs from 'fs/promises';
|
|
2
|
+
import * as path from 'path';
|
|
3
|
+
import { logger } from './logger.js';
|
|
4
|
+
/**
|
|
5
|
+
* Cross-platform file operations utility
|
|
6
|
+
* Centralizes common file operations with progress reporting and error handling
|
|
7
|
+
*/
|
|
8
|
+
export class FileOperations {
|
|
9
|
+
/**
|
|
10
|
+
* Recursively copy a directory with progress reporting
|
|
11
|
+
* Works cross-platform without relying on shell commands
|
|
12
|
+
*/
|
|
13
|
+
static async copyDirectory(src, dest, options = {}) {
|
|
14
|
+
const { onProgress, excludePatterns = [], maxRetries = 3 } = options;
|
|
15
|
+
// First, calculate total files for progress reporting
|
|
16
|
+
const stats = await this.calculateDirectoryStats(src, excludePatterns);
|
|
17
|
+
let copiedFiles = 0;
|
|
18
|
+
await this.copyDirectoryRecursive(src, dest, excludePatterns, maxRetries, (currentFile) => {
|
|
19
|
+
copiedFiles++;
|
|
20
|
+
if (onProgress) {
|
|
21
|
+
onProgress(copiedFiles, stats.totalFiles, currentFile);
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Calculate directory statistics for progress reporting
|
|
27
|
+
*/
|
|
28
|
+
static async calculateDirectoryStats(dir, excludePatterns) {
|
|
29
|
+
let totalFiles = 0;
|
|
30
|
+
let totalSize = 0;
|
|
31
|
+
try {
|
|
32
|
+
const entries = await fs.readdir(dir, { withFileTypes: true });
|
|
33
|
+
for (const entry of entries) {
|
|
34
|
+
const fullPath = path.join(dir, entry.name);
|
|
35
|
+
// Skip excluded patterns
|
|
36
|
+
if (this.shouldExclude(entry.name, excludePatterns)) {
|
|
37
|
+
continue;
|
|
38
|
+
}
|
|
39
|
+
if (entry.isDirectory()) {
|
|
40
|
+
const subStats = await this.calculateDirectoryStats(fullPath, excludePatterns);
|
|
41
|
+
totalFiles += subStats.totalFiles;
|
|
42
|
+
totalSize += subStats.totalSize;
|
|
43
|
+
}
|
|
44
|
+
else {
|
|
45
|
+
totalFiles++;
|
|
46
|
+
try {
|
|
47
|
+
const stat = await fs.stat(fullPath);
|
|
48
|
+
totalSize += stat.size;
|
|
49
|
+
}
|
|
50
|
+
catch {
|
|
51
|
+
// Ignore stat errors
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
catch (error) {
|
|
57
|
+
logger.warn(`[FileOperations] Error calculating stats for ${dir}:`, error);
|
|
58
|
+
}
|
|
59
|
+
return { totalFiles, totalSize };
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Internal recursive copy implementation
|
|
63
|
+
*/
|
|
64
|
+
static async copyDirectoryRecursive(src, dest, excludePatterns, maxRetries, onFileCopied) {
|
|
65
|
+
// Ensure destination directory exists
|
|
66
|
+
await fs.mkdir(dest, { recursive: true });
|
|
67
|
+
const entries = await fs.readdir(src, { withFileTypes: true });
|
|
68
|
+
for (const entry of entries) {
|
|
69
|
+
const srcPath = path.join(src, entry.name);
|
|
70
|
+
const destPath = path.join(dest, entry.name);
|
|
71
|
+
// Skip excluded patterns
|
|
72
|
+
if (this.shouldExclude(entry.name, excludePatterns)) {
|
|
73
|
+
logger.debug(`[FileOperations] Skipping excluded: ${entry.name}`);
|
|
74
|
+
continue;
|
|
75
|
+
}
|
|
76
|
+
if (entry.isDirectory()) {
|
|
77
|
+
await this.copyDirectoryRecursive(srcPath, destPath, excludePatterns, maxRetries, onFileCopied);
|
|
78
|
+
}
|
|
79
|
+
else {
|
|
80
|
+
await this.copyFileWithRetry(srcPath, destPath, maxRetries);
|
|
81
|
+
onFileCopied(srcPath);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Copy a single file with retry logic
|
|
87
|
+
*/
|
|
88
|
+
static async copyFileWithRetry(src, dest, maxRetries) {
|
|
89
|
+
let lastError = null;
|
|
90
|
+
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
|
91
|
+
try {
|
|
92
|
+
await fs.copyFile(src, dest);
|
|
93
|
+
return; // Success
|
|
94
|
+
}
|
|
95
|
+
catch (error) {
|
|
96
|
+
lastError = error;
|
|
97
|
+
logger.debug(`[FileOperations] Copy attempt ${attempt} failed for ${src}: ${error}`);
|
|
98
|
+
if (attempt < maxRetries) {
|
|
99
|
+
// Wait before retry (exponential backoff)
|
|
100
|
+
await new Promise(resolve => setTimeout(resolve, Math.pow(2, attempt) * 100));
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
// All retries failed
|
|
105
|
+
throw new Error(`Failed to copy ${src} after ${maxRetries} attempts: ${lastError?.message}`);
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Check if a file/directory should be excluded
|
|
109
|
+
*/
|
|
110
|
+
static shouldExclude(name, patterns) {
|
|
111
|
+
for (const pattern of patterns) {
|
|
112
|
+
if (pattern.includes('*')) {
|
|
113
|
+
// Simple glob support
|
|
114
|
+
const regex = new RegExp('^' + pattern.replace(/\*/g, '.*') + '$');
|
|
115
|
+
if (regex.test(name))
|
|
116
|
+
return true;
|
|
117
|
+
}
|
|
118
|
+
else if (name === pattern) {
|
|
119
|
+
return true;
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
return false;
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Remove a directory with progress reporting
|
|
126
|
+
*/
|
|
127
|
+
static async removeDirectory(dir, options = {}) {
|
|
128
|
+
const stats = await this.calculateDirectoryStats(dir, []);
|
|
129
|
+
let removedFiles = 0;
|
|
130
|
+
await this.removeDirectoryRecursive(dir, () => {
|
|
131
|
+
removedFiles++;
|
|
132
|
+
if (options.onProgress) {
|
|
133
|
+
options.onProgress(removedFiles, stats.totalFiles);
|
|
134
|
+
}
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Internal recursive remove implementation
|
|
139
|
+
*/
|
|
140
|
+
static async removeDirectoryRecursive(dir, onFileRemoved) {
|
|
141
|
+
try {
|
|
142
|
+
const entries = await fs.readdir(dir, { withFileTypes: true });
|
|
143
|
+
for (const entry of entries) {
|
|
144
|
+
const fullPath = path.join(dir, entry.name);
|
|
145
|
+
if (entry.isDirectory()) {
|
|
146
|
+
await this.removeDirectoryRecursive(fullPath, onFileRemoved);
|
|
147
|
+
}
|
|
148
|
+
else {
|
|
149
|
+
await fs.unlink(fullPath);
|
|
150
|
+
onFileRemoved();
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
// Remove the now-empty directory
|
|
154
|
+
await fs.rmdir(dir);
|
|
155
|
+
}
|
|
156
|
+
catch (error) {
|
|
157
|
+
logger.error(`[FileOperations] Error removing directory ${dir}:`, error);
|
|
158
|
+
throw error;
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* Create a transaction manager for atomic file operations
|
|
163
|
+
*/
|
|
164
|
+
static createTransaction() {
|
|
165
|
+
return new FileTransaction();
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* Transaction manager for atomic file operations
|
|
170
|
+
* Ensures all operations succeed or all are rolled back
|
|
171
|
+
*/
|
|
172
|
+
export class FileTransaction {
|
|
173
|
+
operations = [];
|
|
174
|
+
completed = false;
|
|
175
|
+
/**
|
|
176
|
+
* Add a move operation to the transaction
|
|
177
|
+
*/
|
|
178
|
+
async addMove(source, destination) {
|
|
179
|
+
if (this.completed) {
|
|
180
|
+
throw new Error('Transaction already completed');
|
|
181
|
+
}
|
|
182
|
+
// Perform the move
|
|
183
|
+
await fs.rename(source, destination);
|
|
184
|
+
// Add rollback operation
|
|
185
|
+
this.operations.push({
|
|
186
|
+
type: 'move',
|
|
187
|
+
source,
|
|
188
|
+
destination,
|
|
189
|
+
rollback: async () => {
|
|
190
|
+
try {
|
|
191
|
+
await fs.rename(destination, source);
|
|
192
|
+
}
|
|
193
|
+
catch (error) {
|
|
194
|
+
logger.error(`[FileTransaction] Failed to rollback move from ${destination} to ${source}:`, error);
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
});
|
|
198
|
+
}
|
|
199
|
+
/**
|
|
200
|
+
* Add a copy operation to the transaction
|
|
201
|
+
*/
|
|
202
|
+
async addCopy(source, destination) {
|
|
203
|
+
if (this.completed) {
|
|
204
|
+
throw new Error('Transaction already completed');
|
|
205
|
+
}
|
|
206
|
+
// Perform the copy
|
|
207
|
+
await FileOperations.copyDirectory(source, destination);
|
|
208
|
+
// Add rollback operation
|
|
209
|
+
this.operations.push({
|
|
210
|
+
type: 'copy',
|
|
211
|
+
source,
|
|
212
|
+
destination,
|
|
213
|
+
rollback: async () => {
|
|
214
|
+
try {
|
|
215
|
+
await fs.rm(destination, { recursive: true, force: true });
|
|
216
|
+
}
|
|
217
|
+
catch (error) {
|
|
218
|
+
logger.error(`[FileTransaction] Failed to rollback copy at ${destination}:`, error);
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
});
|
|
222
|
+
}
|
|
223
|
+
/**
|
|
224
|
+
* Add a delete operation to the transaction
|
|
225
|
+
*/
|
|
226
|
+
async addDelete(path, backupPath) {
|
|
227
|
+
if (this.completed) {
|
|
228
|
+
throw new Error('Transaction already completed');
|
|
229
|
+
}
|
|
230
|
+
// If backup path provided, move instead of delete
|
|
231
|
+
if (backupPath) {
|
|
232
|
+
await fs.rename(path, backupPath);
|
|
233
|
+
this.operations.push({
|
|
234
|
+
type: 'delete',
|
|
235
|
+
source: path,
|
|
236
|
+
destination: backupPath,
|
|
237
|
+
rollback: async () => {
|
|
238
|
+
try {
|
|
239
|
+
await fs.rename(backupPath, path);
|
|
240
|
+
}
|
|
241
|
+
catch (error) {
|
|
242
|
+
logger.error(`[FileTransaction] Failed to restore deleted item from ${backupPath} to ${path}:`, error);
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
});
|
|
246
|
+
}
|
|
247
|
+
else {
|
|
248
|
+
// Direct delete (no rollback possible)
|
|
249
|
+
await fs.rm(path, { recursive: true, force: true });
|
|
250
|
+
this.operations.push({
|
|
251
|
+
type: 'delete',
|
|
252
|
+
source: path,
|
|
253
|
+
rollback: async () => {
|
|
254
|
+
logger.warn(`[FileTransaction] Cannot rollback permanent deletion of ${path}`);
|
|
255
|
+
}
|
|
256
|
+
});
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
/**
|
|
260
|
+
* Commit the transaction (mark as successful)
|
|
261
|
+
*/
|
|
262
|
+
commit() {
|
|
263
|
+
this.completed = true;
|
|
264
|
+
}
|
|
265
|
+
/**
|
|
266
|
+
* Rollback all operations in reverse order
|
|
267
|
+
*/
|
|
268
|
+
async rollback() {
|
|
269
|
+
logger.info(`[FileTransaction] Rolling back ${this.operations.length} operations`);
|
|
270
|
+
// Rollback in reverse order
|
|
271
|
+
for (let i = this.operations.length - 1; i >= 0; i--) {
|
|
272
|
+
const operation = this.operations[i];
|
|
273
|
+
logger.info(`[FileTransaction] Rolling back ${operation.type} operation`);
|
|
274
|
+
try {
|
|
275
|
+
await operation.rollback();
|
|
276
|
+
}
|
|
277
|
+
catch (error) {
|
|
278
|
+
logger.error(`[FileTransaction] Rollback failed for operation ${i}:`, error);
|
|
279
|
+
// Continue with other rollbacks
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
this.completed = true;
|
|
283
|
+
}
|
|
284
|
+
/**
|
|
285
|
+
* Check if any operations have been performed
|
|
286
|
+
*/
|
|
287
|
+
hasOperations() {
|
|
288
|
+
return this.operations.length > 0;
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fileOperations.js","sourceRoot":"","sources":["../../src/utils/fileOperations.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { logger } from './logger.js';\n\nexport interface CopyOptions {\n  onProgress?: (copied: number, total: number, currentFile: string) => void;\n  excludePatterns?: string[];\n  maxRetries?: number;\n}\n\nexport interface FileStats {\n  totalFiles: number;\n  totalSize: number;\n}\n\n/**\n * Cross-platform file operations utility\n * Centralizes common file operations with progress reporting and error handling\n */\nexport class FileOperations {\n  /**\n   * Recursively copy a directory with progress reporting\n   * Works cross-platform without relying on shell commands\n   */\n  static async copyDirectory(\n    src: string, \n    dest: string, \n    options: CopyOptions = {}\n  ): Promise<void> {\n    const { onProgress, excludePatterns = [], maxRetries = 3 } = options;\n    \n    // First, calculate total files for progress reporting\n    const stats = await this.calculateDirectoryStats(src, excludePatterns);\n    let copiedFiles = 0;\n    \n    await this.copyDirectoryRecursive(\n      src, \n      dest, \n      excludePatterns,\n      maxRetries,\n      (currentFile) => {\n        copiedFiles++;\n        if (onProgress) {\n          onProgress(copiedFiles, stats.totalFiles, currentFile);\n        }\n      }\n    );\n  }\n  \n  /**\n   * Calculate directory statistics for progress reporting\n   */\n  private static async calculateDirectoryStats(\n    dir: string,\n    excludePatterns: string[]\n  ): Promise<FileStats> {\n    let totalFiles = 0;\n    let totalSize = 0;\n    \n    try {\n      const entries = await fs.readdir(dir, { withFileTypes: true });\n      \n      for (const entry of entries) {\n        const fullPath = path.join(dir, entry.name);\n        \n        // Skip excluded patterns\n        if (this.shouldExclude(entry.name, excludePatterns)) {\n          continue;\n        }\n        \n        if (entry.isDirectory()) {\n          const subStats = await this.calculateDirectoryStats(fullPath, excludePatterns);\n          totalFiles += subStats.totalFiles;\n          totalSize += subStats.totalSize;\n        } else {\n          totalFiles++;\n          try {\n            const stat = await fs.stat(fullPath);\n            totalSize += stat.size;\n          } catch {\n            // Ignore stat errors\n          }\n        }\n      }\n    } catch (error) {\n      logger.warn(`[FileOperations] Error calculating stats for ${dir}:`, error);\n    }\n    \n    return { totalFiles, totalSize };\n  }\n  \n  /**\n   * Internal recursive copy implementation\n   */\n  private static async copyDirectoryRecursive(\n    src: string,\n    dest: string,\n    excludePatterns: string[],\n    maxRetries: number,\n    onFileCopied: (file: string) => void\n  ): Promise<void> {\n    // Ensure destination directory exists\n    await fs.mkdir(dest, { recursive: true });\n    \n    const entries = await fs.readdir(src, { withFileTypes: true });\n    \n    for (const entry of entries) {\n      const srcPath = path.join(src, entry.name);\n      const destPath = path.join(dest, entry.name);\n      \n      // Skip excluded patterns\n      if (this.shouldExclude(entry.name, excludePatterns)) {\n        logger.debug(`[FileOperations] Skipping excluded: ${entry.name}`);\n        continue;\n      }\n      \n      if (entry.isDirectory()) {\n        await this.copyDirectoryRecursive(\n          srcPath, \n          destPath, \n          excludePatterns,\n          maxRetries,\n          onFileCopied\n        );\n      } else {\n        await this.copyFileWithRetry(srcPath, destPath, maxRetries);\n        onFileCopied(srcPath);\n      }\n    }\n  }\n  \n  /**\n   * Copy a single file with retry logic\n   */\n  private static async copyFileWithRetry(\n    src: string,\n    dest: string,\n    maxRetries: number\n  ): Promise<void> {\n    let lastError: Error | null = null;\n    \n    for (let attempt = 1; attempt <= maxRetries; attempt++) {\n      try {\n        await fs.copyFile(src, dest);\n        return; // Success\n      } catch (error) {\n        lastError = error as Error;\n        logger.debug(`[FileOperations] Copy attempt ${attempt} failed for ${src}: ${error}`);\n        \n        if (attempt < maxRetries) {\n          // Wait before retry (exponential backoff)\n          await new Promise(resolve => setTimeout(resolve, Math.pow(2, attempt) * 100));\n        }\n      }\n    }\n    \n    // All retries failed\n    throw new Error(`Failed to copy ${src} after ${maxRetries} attempts: ${lastError?.message}`);\n  }\n  \n  /**\n   * Check if a file/directory should be excluded\n   */\n  private static shouldExclude(name: string, patterns: string[]): boolean {\n    for (const pattern of patterns) {\n      if (pattern.includes('*')) {\n        // Simple glob support\n        const regex = new RegExp('^' + pattern.replace(/\\*/g, '.*') + '$');\n        if (regex.test(name)) return true;\n      } else if (name === pattern) {\n        return true;\n      }\n    }\n    return false;\n  }\n  \n  /**\n   * Remove a directory with progress reporting\n   */\n  static async removeDirectory(\n    dir: string,\n    options: { onProgress?: (removed: number, total: number) => void } = {}\n  ): Promise<void> {\n    const stats = await this.calculateDirectoryStats(dir, []);\n    let removedFiles = 0;\n    \n    await this.removeDirectoryRecursive(dir, () => {\n      removedFiles++;\n      if (options.onProgress) {\n        options.onProgress(removedFiles, stats.totalFiles);\n      }\n    });\n  }\n  \n  /**\n   * Internal recursive remove implementation\n   */\n  private static async removeDirectoryRecursive(\n    dir: string,\n    onFileRemoved: () => void\n  ): Promise<void> {\n    try {\n      const entries = await fs.readdir(dir, { withFileTypes: true });\n      \n      for (const entry of entries) {\n        const fullPath = path.join(dir, entry.name);\n        \n        if (entry.isDirectory()) {\n          await this.removeDirectoryRecursive(fullPath, onFileRemoved);\n        } else {\n          await fs.unlink(fullPath);\n          onFileRemoved();\n        }\n      }\n      \n      // Remove the now-empty directory\n      await fs.rmdir(dir);\n    } catch (error) {\n      logger.error(`[FileOperations] Error removing directory ${dir}:`, error);\n      throw error;\n    }\n  }\n  \n  /**\n   * Create a transaction manager for atomic file operations\n   */\n  static createTransaction(): FileTransaction {\n    return new FileTransaction();\n  }\n}\n\n/**\n * Transaction manager for atomic file operations\n * Ensures all operations succeed or all are rolled back\n */\nexport class FileTransaction {\n  private operations: Array<{\n    type: 'move' | 'copy' | 'delete' | 'create';\n    source?: string;\n    destination?: string;\n    rollback: () => Promise<void>;\n  }> = [];\n  \n  private completed = false;\n  \n  /**\n   * Add a move operation to the transaction\n   */\n  async addMove(source: string, destination: string): Promise<void> {\n    if (this.completed) {\n      throw new Error('Transaction already completed');\n    }\n    \n    // Perform the move\n    await fs.rename(source, destination);\n    \n    // Add rollback operation\n    this.operations.push({\n      type: 'move',\n      source,\n      destination,\n      rollback: async () => {\n        try {\n          await fs.rename(destination, source);\n        } catch (error) {\n          logger.error(`[FileTransaction] Failed to rollback move from ${destination} to ${source}:`, error);\n        }\n      }\n    });\n  }\n  \n  /**\n   * Add a copy operation to the transaction\n   */\n  async addCopy(source: string, destination: string): Promise<void> {\n    if (this.completed) {\n      throw new Error('Transaction already completed');\n    }\n    \n    // Perform the copy\n    await FileOperations.copyDirectory(source, destination);\n    \n    // Add rollback operation\n    this.operations.push({\n      type: 'copy',\n      source,\n      destination,\n      rollback: async () => {\n        try {\n          await fs.rm(destination, { recursive: true, force: true });\n        } catch (error) {\n          logger.error(`[FileTransaction] Failed to rollback copy at ${destination}:`, error);\n        }\n      }\n    });\n  }\n  \n  /**\n   * Add a delete operation to the transaction\n   */\n  async addDelete(path: string, backupPath?: string): Promise<void> {\n    if (this.completed) {\n      throw new Error('Transaction already completed');\n    }\n    \n    // If backup path provided, move instead of delete\n    if (backupPath) {\n      await fs.rename(path, backupPath);\n      \n      this.operations.push({\n        type: 'delete',\n        source: path,\n        destination: backupPath,\n        rollback: async () => {\n          try {\n            await fs.rename(backupPath, path);\n          } catch (error) {\n            logger.error(`[FileTransaction] Failed to restore deleted item from ${backupPath} to ${path}:`, error);\n          }\n        }\n      });\n    } else {\n      // Direct delete (no rollback possible)\n      await fs.rm(path, { recursive: true, force: true });\n      \n      this.operations.push({\n        type: 'delete',\n        source: path,\n        rollback: async () => {\n          logger.warn(`[FileTransaction] Cannot rollback permanent deletion of ${path}`);\n        }\n      });\n    }\n  }\n  \n  /**\n   * Commit the transaction (mark as successful)\n   */\n  commit(): void {\n    this.completed = true;\n  }\n  \n  /**\n   * Rollback all operations in reverse order\n   */\n  async rollback(): Promise<void> {\n    logger.info(`[FileTransaction] Rolling back ${this.operations.length} operations`);\n    \n    // Rollback in reverse order\n    for (let i = this.operations.length - 1; i >= 0; i--) {\n      const operation = this.operations[i];\n      logger.info(`[FileTransaction] Rolling back ${operation.type} operation`);\n      \n      try {\n        await operation.rollback();\n      } catch (error) {\n        logger.error(`[FileTransaction] Rollback failed for operation ${i}:`, error);\n        // Continue with other rollbacks\n      }\n    }\n    \n    this.completed = true;\n  }\n  \n  /**\n   * Check if any operations have been performed\n   */\n  hasOperations(): boolean {\n    return this.operations.length > 0;\n  }\n}"]}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Installation detection utilities
|
|
3
|
+
* Determines whether the application is running from npm or git installation
|
|
4
|
+
*/
|
|
5
|
+
export type InstallationType = 'npm' | 'git' | 'unknown';
|
|
6
|
+
export declare class InstallationDetector {
|
|
7
|
+
private static cachedType;
|
|
8
|
+
private static get MAX_SEARCH_DEPTH();
|
|
9
|
+
/**
|
|
10
|
+
* Detect the installation type (npm global, git clone, or unknown)
|
|
11
|
+
*/
|
|
12
|
+
static getInstallationType(): InstallationType;
|
|
13
|
+
/**
|
|
14
|
+
* Get the npm global installation path if running from npm
|
|
15
|
+
*/
|
|
16
|
+
static getNpmGlobalPath(): string | null;
|
|
17
|
+
/**
|
|
18
|
+
* Get the git repository root path if running from git
|
|
19
|
+
*/
|
|
20
|
+
static getGitRepositoryPath(): string | null;
|
|
21
|
+
/**
|
|
22
|
+
* Get a human-readable description of the installation
|
|
23
|
+
*/
|
|
24
|
+
static getInstallationDescription(): string;
|
|
25
|
+
/**
|
|
26
|
+
* Clear the cached installation type (mainly for testing)
|
|
27
|
+
*/
|
|
28
|
+
static clearCache(): void;
|
|
29
|
+
}
|
|
30
|
+
//# sourceMappingURL=installation.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"installation.d.ts","sourceRoot":"","sources":["../../src/utils/installation.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAQH,MAAM,MAAM,gBAAgB,GAAG,KAAK,GAAG,KAAK,GAAG,SAAS,CAAC;AAEzD,qBAAa,oBAAoB;IAC/B,OAAO,CAAC,MAAM,CAAC,UAAU,CAAiC;IAG1D,OAAO,CAAC,MAAM,KAAK,gBAAgB,GAElC;IAED;;OAEG;IACH,MAAM,CAAC,mBAAmB,IAAI,gBAAgB;IA8D9C;;OAEG;IACH,MAAM,CAAC,gBAAgB,IAAI,MAAM,GAAG,IAAI;IA2BxC;;OAEG;IACH,MAAM,CAAC,oBAAoB,IAAI,MAAM,GAAG,IAAI;IA4B5C;;OAEG;IACH,MAAM,CAAC,0BAA0B,IAAI,MAAM;IAqB3C;;OAEG;IACH,MAAM,CAAC,UAAU,IAAI,IAAI;CAG1B"}
|