Files
markitect-main/capabilities/release-management/src/release_management/utils/version.py
tegwick d0ffdc057c feat: implement modular capability system with automatic discovery
- Move release management to capabilities/release-management/ with complete Makefile
- Create automatic capability discovery system in scripts/capability_discovery.mk
- Add capability-manager subagent for managing modular architecture
- Implement target delegation system enabling capability-name-target patterns
- Create Makefiles for markitect-content, markitect-utils, and issue-facade capabilities
- Remove legacy release management code and documentation from main project
- Update main Makefile to use capability discovery and delegation
- Add comprehensive capability status, help, and management targets

The capability system provides:
- Automatic discovery of capabilities with Makefiles
- Clean target delegation without conflicts
- Modular architecture following established patterns
- Comprehensive help and status reporting
- Zero-conflict capability integration

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-09 01:29:15 +01:00

191 lines
5.9 KiB
Python

"""
Version management utilities.
This module provides utilities for working with versions and setuptools-scm.
"""
import subprocess
from pathlib import Path
from typing import Optional, Dict, Any
from packaging import version
class VersionManager:
"""Utilities for version management with setuptools-scm."""
def __init__(self, project_root: Optional[Path] = None):
"""Initialize version manager.
Args:
project_root: Root directory of the project
"""
self.project_root = project_root or Path.cwd()
def get_current_version(self) -> str:
"""Get current version using setuptools-scm.
Returns:
Current version string or "unknown" if unavailable
"""
try:
result = subprocess.run(
['python', '-m', 'setuptools_scm'],
capture_output=True,
text=True,
check=True,
cwd=self.project_root
)
return result.stdout.strip()
except subprocess.CalledProcessError:
return "unknown"
def parse_version(self, version_string: str) -> Dict[str, Any]:
"""Parse a version string and return components.
Args:
version_string: Version string to parse
Returns:
Dictionary with version components
"""
try:
v = version.Version(version_string)
return {
'major': v.major,
'minor': v.minor,
'micro': v.micro,
'is_prerelease': v.is_prerelease,
'is_devrelease': v.is_devrelease,
'local': v.local,
'public': v.public,
'base_version': v.base_version,
}
except version.InvalidVersion:
return {'error': f'Invalid version: {version_string}'}
def is_development_version(self, version_string: Optional[str] = None) -> bool:
"""Check if version is a development version.
Args:
version_string: Version to check. If None, uses current version.
Returns:
True if development version, False otherwise
"""
if version_string is None:
version_string = self.get_current_version()
try:
v = version.Version(version_string)
return v.is_devrelease or 'dev' in version_string.lower()
except version.InvalidVersion:
return True # Assume unknown versions are dev
def compare_versions(self, version1: str, version2: str) -> int:
"""Compare two version strings.
Args:
version1: First version to compare
version2: Second version to compare
Returns:
-1 if version1 < version2, 0 if equal, 1 if version1 > version2
"""
try:
v1 = version.Version(version1)
v2 = version.Version(version2)
if v1 < v2:
return -1
elif v1 > v2:
return 1
else:
return 0
except version.InvalidVersion:
# Fallback to string comparison
if version1 < version2:
return -1
elif version1 > version2:
return 1
else:
return 0
def get_next_version(self, current_version: str, bump_type: str = 'patch') -> str:
"""Get the next version based on bump type.
Args:
current_version: Current version string
bump_type: Type of bump ('major', 'minor', 'patch')
Returns:
Next version string
Raises:
ValueError: If bump_type is invalid
"""
try:
v = version.Version(current_version)
major, minor, micro = v.major, v.minor, v.micro
if bump_type == 'major':
return f"{major + 1}.0.0"
elif bump_type == 'minor':
return f"{major}.{minor + 1}.0"
elif bump_type == 'patch':
return f"{major}.{minor}.{micro + 1}"
else:
raise ValueError(f"Invalid bump type: {bump_type}")
except version.InvalidVersion:
raise ValueError(f"Cannot parse version: {current_version}")
def suggest_version(self, current_version: Optional[str] = None) -> Dict[str, str]:
"""Suggest next version options.
Args:
current_version: Current version. If None, gets from setuptools-scm.
Returns:
Dictionary with version suggestions
"""
if current_version is None:
current_version = self.get_current_version()
if current_version == "unknown":
return {
'error': 'Cannot determine current version',
'suggestion': 'Consider creating an initial tag like v0.1.0'
}
try:
# Strip development version info to get base
v = version.Version(current_version)
base_version = v.base_version
return {
'current': current_version,
'base': base_version,
'patch': self.get_next_version(base_version, 'patch'),
'minor': self.get_next_version(base_version, 'minor'),
'major': self.get_next_version(base_version, 'major'),
}
except Exception as e:
return {
'error': str(e),
'current': current_version
}
def validate_version_format(self, version_string: str) -> bool:
"""Validate if a version string follows semantic versioning.
Args:
version_string: Version string to validate
Returns:
True if valid semantic version, False otherwise
"""
try:
version.Version(version_string)
return True
except version.InvalidVersion:
return False