@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.
Files changed (42) hide show
  1. package/README.md +54 -9
  2. package/dist/config/updateConfig.d.ts +84 -0
  3. package/dist/config/updateConfig.d.ts.map +1 -0
  4. package/dist/config/updateConfig.js +148 -0
  5. package/dist/generated/version.d.ts +9 -0
  6. package/dist/generated/version.d.ts.map +1 -0
  7. package/dist/generated/version.js +9 -0
  8. package/dist/index.d.ts +6 -0
  9. package/dist/index.d.ts.map +1 -1
  10. package/dist/index.js +34 -6
  11. package/dist/portfolio/DefaultElementProvider.d.ts +78 -0
  12. package/dist/portfolio/DefaultElementProvider.d.ts.map +1 -0
  13. package/dist/portfolio/DefaultElementProvider.js +398 -0
  14. package/dist/portfolio/PortfolioManager.d.ts +7 -0
  15. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  16. package/dist/portfolio/PortfolioManager.js +44 -3
  17. package/dist/security/commandValidator.d.ts.map +1 -1
  18. package/dist/security/commandValidator.js +5 -2
  19. package/dist/security/securityMonitor.d.ts +2 -1
  20. package/dist/security/securityMonitor.d.ts.map +1 -1
  21. package/dist/security/securityMonitor.js +1 -1
  22. package/dist/server/tools/UpdateTools.d.ts.map +1 -1
  23. package/dist/server/tools/UpdateTools.js +22 -1
  24. package/dist/server/types.d.ts +1 -0
  25. package/dist/server/types.d.ts.map +1 -1
  26. package/dist/server/types.js +1 -1
  27. package/dist/update/BackupManager.d.ts +17 -0
  28. package/dist/update/BackupManager.d.ts.map +1 -1
  29. package/dist/update/BackupManager.js +112 -3
  30. package/dist/update/UpdateManager.d.ts +19 -0
  31. package/dist/update/UpdateManager.d.ts.map +1 -1
  32. package/dist/update/UpdateManager.js +485 -15
  33. package/dist/update/VersionManager.d.ts +1 -1
  34. package/dist/update/VersionManager.d.ts.map +1 -1
  35. package/dist/update/VersionManager.js +62 -15
  36. package/dist/utils/fileOperations.d.ts +83 -0
  37. package/dist/utils/fileOperations.d.ts.map +1 -0
  38. package/dist/utils/fileOperations.js +291 -0
  39. package/dist/utils/installation.d.ts +30 -0
  40. package/dist/utils/installation.d.ts.map +1 -0
  41. package/dist/utils/installation.js +160 -0
  42. package/package.json +3 -1
@@ -3,7 +3,7 @@
3
3
  */
4
4
  export declare class VersionManager {
5
5
  /**
6
- * Get current version from package.json
6
+ * Get current version from package.json or embedded version
7
7
  */
8
8
  getCurrentVersion(): Promise<string>;
9
9
  /**
@@ -1 +1 @@
1
- {"version":3,"file":"VersionManager.d.ts","sourceRoot":"","sources":["../../src/update/VersionManager.ts"],"names":[],"mappings":"AAAA;;GAEG;AAKH,qBAAa,cAAc;IACzB;;OAEG;IACG,iBAAiB,IAAI,OAAO,CAAC,MAAM,CAAC;IAgC1C;;;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"}
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
- // Use process.cwd() as a base, then search upward for package.json
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
- packageJsonPath = candidatePath;
20
- break;
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
- if (!packageJsonPath) {
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":"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","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"}