- 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>
191 lines
5.9 KiB
Python
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 |