@mickdarling/dollhousemcp
Version:
DollhouseMCP - A Model Context Protocol (MCP) server that enables dynamic AI persona management from markdown files, allowing Claude and other compatible AI assistants to activate and switch between different behavioral personas.
97 lines • 12.7 kB
JavaScript
/**
* Version comparison and parsing utilities
*/
/**
* Compare two semantic versions
* @returns -1 if version1 < version2, 0 if equal, 1 if version1 > version2
*/
export function compareVersions(version1, version2) {
// Normalize versions by removing 'v' prefix
const v1 = version1.replace(/^v/, '');
const v2 = version2.replace(/^v/, '');
// Split version and pre-release parts
const [v1main, v1pre] = v1.split('-');
const [v2main, v2pre] = v2.split('-');
// Compare main version parts (x.y.z)
const v1parts = v1main.split('.').map(part => parseInt(part) || 0);
const v2parts = v2main.split('.').map(part => parseInt(part) || 0);
const maxLength = Math.max(v1parts.length, v2parts.length);
for (let i = 0; i < maxLength; i++) {
const v1part = v1parts[i] || 0;
const v2part = v2parts[i] || 0;
if (v1part < v2part)
return -1;
if (v1part > v2part)
return 1;
}
// If main versions are equal, compare pre-release versions
// Version without pre-release is greater than version with pre-release
if (!v1pre && v2pre)
return 1; // 1.0.0 > 1.0.0-beta
if (v1pre && !v2pre)
return -1; // 1.0.0-beta < 1.0.0
if (!v1pre && !v2pre)
return 0; // 1.0.0 == 1.0.0
// Both have pre-release, compare lexicographically
return v1pre.localeCompare(v2pre);
}
/**
* Parse version from command output
*/
export function parseVersionFromOutput(output, tool) {
// Git version output: "git version 2.39.2"
// npm version output: "8.19.2" or JSON with version info
if (tool === 'git') {
const match = output.match(/git version (\d+\.\d+\.\d+)/);
return match ? match[1] : null;
}
else if (tool === 'npm') {
// npm might return just the version number or JSON
const cleanOutput = output.trim();
if (cleanOutput.match(/^\d+\.\d+\.\d+/)) {
return cleanOutput.split('\n')[0]; // First line if multiple lines
}
// Try to parse as JSON if it looks like JSON
try {
const parsed = JSON.parse(cleanOutput);
return parsed.npm || parsed.version || null;
}
catch {
// If not JSON, try to extract version pattern
const match = cleanOutput.match(/(\d+\.\d+\.\d+)/);
return match ? match[1] : null;
}
}
return null;
}
/**
* Validate dependency version against requirements
*/
export function validateDependencyVersion(actualVersion, requirements, toolName) {
const minComparison = compareVersions(actualVersion, requirements.minimum);
const maxComparison = compareVersions(actualVersion, requirements.maximum);
// Check minimum version requirement
if (minComparison < 0) {
return {
valid: false,
error: `${toolName} version ${actualVersion} is below minimum required version ${requirements.minimum}`
};
}
// Check maximum version requirement
if (maxComparison > 0) {
return {
valid: false,
error: `${toolName} version ${actualVersion} is above maximum tested version ${requirements.maximum}`
};
}
// Check if it's the recommended version
const recComparison = compareVersions(actualVersion, requirements.recommended);
if (recComparison !== 0) {
return {
valid: true,
warning: `${toolName} version ${actualVersion} works but ${requirements.recommended} is recommended for optimal stability`
};
}
return { valid: true };
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"version.js","sourceRoot":"","sources":["../../src/utils/version.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Version comparison and parsing utilities\n */\n\n/**\n * Compare two semantic versions\n * @returns -1 if version1 < version2, 0 if equal, 1 if version1 > version2\n */\nexport function 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 command output\n */\nexport function 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 dependency version against requirements\n */\nexport function 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 = compareVersions(actualVersion, requirements.minimum);\n  const maxComparison = compareVersions(actualVersion, requirements.maximum);\n  \n  // Check minimum version requirement\n  if (minComparison < 0) {\n    return {\n      valid: false,\n      error: `${toolName} version ${actualVersion} is below minimum required version ${requirements.minimum}`\n    };\n  }\n  \n  // Check maximum version requirement\n  if (maxComparison > 0) {\n    return {\n      valid: false,\n      error: `${toolName} version ${actualVersion} is above maximum tested version ${requirements.maximum}`\n    };\n  }\n  \n  // Check if it's the recommended version\n  const recComparison = 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  return { valid: true };\n}"]}