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>
This commit is contained in:
2025-11-09 01:29:15 +01:00
parent d505c15d40
commit d0ffdc057c
38 changed files with 3978 additions and 1361 deletions

View File

@@ -0,0 +1,205 @@
"""
Git operations for release management.
This module handles all Git-related operations needed for releases.
"""
import subprocess
from pathlib import Path
from typing import Dict, Any, Optional, List
class GitManager:
"""Manages Git operations for releases."""
def __init__(self, project_root: Optional[Path] = None, dry_run: bool = False):
"""Initialize Git manager.
Args:
project_root: Root directory of the project
dry_run: If True, show what would be done without executing
"""
self.project_root = project_root or Path.cwd()
self.dry_run = dry_run
def get_repository_status(self) -> Dict[str, Any]:
"""Get current git repository status.
Returns:
Dictionary with repository status information
"""
try:
# Get current branch
branch_result = self._run_command(['git', 'branch', '--show-current'])
current_branch = branch_result.stdout.strip()
# Check for uncommitted changes
status_result = self._run_command(['git', 'status', '--porcelain'])
has_changes = bool(status_result.stdout.strip())
# Get latest commit
commit_result = self._run_command(['git', 'rev-parse', '--short', 'HEAD'])
latest_commit = commit_result.stdout.strip()
# Get latest tag
try:
tag_result = self._run_command(['git', 'describe', '--tags', '--abbrev=0'])
latest_tag = tag_result.stdout.strip()
except subprocess.CalledProcessError:
latest_tag = None
return {
'is_repo': True,
'branch': current_branch,
'has_changes': has_changes,
'latest_commit': latest_commit,
'latest_tag': latest_tag
}
except subprocess.CalledProcessError:
return {'is_repo': False}
def create_tag(self, version: str, message: Optional[str] = None) -> bool:
"""Create and push git tag.
Args:
version: Version to tag (e.g., "1.0.0")
message: Optional tag message
Returns:
True if successful, False otherwise
"""
if not version.startswith('v'):
tag_name = f"v{version}"
else:
tag_name = version
tag_message = message or f"Release {version.lstrip('v')}"
print(f"🏷️ Creating git tag {tag_name}")
try:
# Create annotated tag
self._run_command(['git', 'tag', '-a', tag_name, '-m', tag_message])
print(f"✅ Tag {tag_name} created")
# Push tag to origin
try:
print(f"📤 Pushing tag to origin...")
self._run_command(['git', 'push', 'origin', tag_name])
print(f"✅ Tag pushed to origin")
return True
except subprocess.CalledProcessError as e:
print(f"⚠️ Could not push tag to origin: {e}")
print(f"You can push it manually with: git push origin {tag_name}")
return True # Tag created successfully, push can be done manually
except subprocess.CalledProcessError as e:
print(f"❌ Failed to create tag: {e}")
return False
def validate_release_state(self, force: bool = False) -> tuple[bool, List[str]]:
"""Validate that repository is ready for release.
Args:
force: Skip validation checks if True
Returns:
Tuple of (is_valid, list_of_issues)
"""
issues = []
status = self.get_repository_status()
if not status['is_repo']:
issues.append("Not in a git repository")
else:
if status['has_changes'] and not force:
issues.append("Repository has uncommitted changes")
if status['branch'] != 'main' and not force:
issues.append(f"Not on main branch (currently on {status['branch']})")
return len(issues) == 0, issues
def get_commits_since_tag(self, tag_name: Optional[str] = None) -> List[str]:
"""Get list of commits since specified tag.
Args:
tag_name: Tag to compare against. If None, uses latest tag.
Returns:
List of commit messages since tag
"""
try:
if tag_name is None:
# Get latest tag
tag_result = self._run_command(['git', 'describe', '--tags', '--abbrev=0'])
tag_name = tag_result.stdout.strip()
# Get commits since tag
log_result = self._run_command([
'git', 'log', f'{tag_name}..HEAD', '--oneline', '--no-merges'
])
commits = []
for line in log_result.stdout.strip().split('\n'):
if line:
commits.append(line)
return commits
except subprocess.CalledProcessError:
return []
def tag_exists(self, tag_name: str) -> bool:
"""Check if a git tag exists.
Args:
tag_name: Tag name to check
Returns:
True if tag exists, False otherwise
"""
try:
self._run_command(['git', 'rev-parse', f'refs/tags/{tag_name}'])
return True
except subprocess.CalledProcessError:
return False
def get_remote_url(self, remote: str = 'origin') -> Optional[str]:
"""Get the URL of a git remote.
Args:
remote: Remote name (default: 'origin')
Returns:
Remote URL or None if not found
"""
try:
result = self._run_command(['git', 'remote', 'get-url', remote])
return result.stdout.strip()
except subprocess.CalledProcessError:
return None
def _run_command(self, cmd: List[str]) -> subprocess.CompletedProcess:
"""Run a git command.
Args:
cmd: Command to execute
Returns:
CompletedProcess result
Raises:
subprocess.CalledProcessError: If command fails
"""
if self.dry_run and not any(read_only in cmd for read_only in
['status', 'branch', 'rev-parse', 'describe',
'log', 'remote']):
print(f"[DRY RUN] Would run: {' '.join(cmd)}")
return subprocess.CompletedProcess(cmd, 0, "", "")
return subprocess.run(
cmd,
capture_output=True,
text=True,
check=True,
cwd=self.project_root
)