claude-mpm 4.2.44__py3-none-any.whl → 4.3.0__py3-none-any.whl
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.
- claude_mpm/VERSION +1 -1
- claude_mpm/agents/BASE_PM.md +77 -405
- claude_mpm/agents/{INSTRUCTIONS.md → INSTRUCTIONS_OLD_DEPRECATED.md} +75 -1
- claude_mpm/agents/OUTPUT_STYLE.md +0 -39
- claude_mpm/agents/PM_INSTRUCTIONS.md +122 -0
- claude_mpm/agents/WORKFLOW.md +74 -323
- claude_mpm/agents/frontmatter_validator.py +20 -12
- claude_mpm/agents/templates/nextjs_engineer.json +277 -0
- claude_mpm/agents/templates/prompt-engineer.json +294 -0
- claude_mpm/agents/templates/python_engineer.json +289 -0
- claude_mpm/agents/templates/react_engineer.json +11 -3
- claude_mpm/agents/templates/security.json +50 -9
- claude_mpm/cli/commands/agents.py +2 -2
- claude_mpm/cli/commands/uninstall.py +1 -3
- claude_mpm/cli/interactive/agent_wizard.py +3 -3
- claude_mpm/cli/parsers/agent_manager_parser.py +3 -3
- claude_mpm/cli/parsers/agents_parser.py +1 -1
- claude_mpm/constants.py +1 -1
- claude_mpm/core/error_handler.py +2 -4
- claude_mpm/core/file_utils.py +4 -12
- claude_mpm/core/framework_loader.py +72 -24
- claude_mpm/core/log_manager.py +60 -5
- claude_mpm/core/logger.py +1 -1
- claude_mpm/core/logging_utils.py +36 -18
- claude_mpm/core/unified_agent_registry.py +18 -4
- claude_mpm/dashboard/react/components/DataInspector/DataInspector.module.css +188 -0
- claude_mpm/dashboard/react/components/EventViewer/EventViewer.module.css +156 -0
- claude_mpm/dashboard/react/components/shared/ConnectionStatus.module.css +38 -0
- claude_mpm/dashboard/react/components/shared/FilterBar.module.css +92 -0
- claude_mpm/dashboard/static/archive/activity_dashboard_fixed.html +248 -0
- claude_mpm/dashboard/static/archive/activity_dashboard_test.html +61 -0
- claude_mpm/dashboard/static/archive/test_activity_connection.html +179 -0
- claude_mpm/dashboard/static/archive/test_claude_tree_tab.html +68 -0
- claude_mpm/dashboard/static/archive/test_dashboard.html +409 -0
- claude_mpm/dashboard/static/archive/test_dashboard_fixed.html +519 -0
- claude_mpm/dashboard/static/archive/test_dashboard_verification.html +181 -0
- claude_mpm/dashboard/static/archive/test_file_data.html +315 -0
- claude_mpm/dashboard/static/archive/test_file_tree_empty_state.html +243 -0
- claude_mpm/dashboard/static/archive/test_file_tree_fix.html +234 -0
- claude_mpm/dashboard/static/archive/test_file_tree_rename.html +117 -0
- claude_mpm/dashboard/static/archive/test_file_tree_tab.html +115 -0
- claude_mpm/dashboard/static/archive/test_file_viewer.html +224 -0
- claude_mpm/dashboard/static/archive/test_final_activity.html +220 -0
- claude_mpm/dashboard/static/archive/test_tab_fix.html +139 -0
- claude_mpm/dashboard/static/built/assets/events.DjpNxWNo.css +1 -0
- claude_mpm/dashboard/static/built/components/activity-tree.js +1 -1
- claude_mpm/dashboard/static/built/components/agent-hierarchy.js +777 -0
- claude_mpm/dashboard/static/built/components/agent-inference.js +1 -1
- claude_mpm/dashboard/static/built/components/build-tracker.js +333 -0
- claude_mpm/dashboard/static/built/components/code-simple.js +857 -0
- claude_mpm/dashboard/static/built/components/code-tree/tree-breadcrumb.js +353 -0
- claude_mpm/dashboard/static/built/components/code-tree/tree-constants.js +235 -0
- claude_mpm/dashboard/static/built/components/code-tree/tree-search.js +409 -0
- claude_mpm/dashboard/static/built/components/code-tree/tree-utils.js +435 -0
- claude_mpm/dashboard/static/built/components/code-viewer.js +2 -1076
- claude_mpm/dashboard/static/built/components/connection-debug.js +654 -0
- claude_mpm/dashboard/static/built/components/diff-viewer.js +891 -0
- claude_mpm/dashboard/static/built/components/event-processor.js +1 -1
- claude_mpm/dashboard/static/built/components/event-viewer.js +1 -1
- claude_mpm/dashboard/static/built/components/export-manager.js +1 -1
- claude_mpm/dashboard/static/built/components/file-change-tracker.js +443 -0
- claude_mpm/dashboard/static/built/components/file-change-viewer.js +690 -0
- claude_mpm/dashboard/static/built/components/file-tool-tracker.js +1 -1
- claude_mpm/dashboard/static/built/components/module-viewer.js +1 -1
- claude_mpm/dashboard/static/built/components/nav-bar.js +145 -0
- claude_mpm/dashboard/static/built/components/page-structure.js +429 -0
- claude_mpm/dashboard/static/built/components/session-manager.js +1 -1
- claude_mpm/dashboard/static/built/components/ui-state-manager.js +2 -465
- claude_mpm/dashboard/static/built/components/working-directory.js +1 -1
- claude_mpm/dashboard/static/built/connection-manager.js +536 -0
- claude_mpm/dashboard/static/built/dashboard.js +1 -1
- claude_mpm/dashboard/static/built/extension-error-handler.js +164 -0
- claude_mpm/dashboard/static/built/react/events.js +30 -0
- claude_mpm/dashboard/static/built/shared/dom-helpers.js +396 -0
- claude_mpm/dashboard/static/built/shared/event-bus.js +330 -0
- claude_mpm/dashboard/static/built/shared/event-filter-service.js +540 -0
- claude_mpm/dashboard/static/built/shared/logger.js +385 -0
- claude_mpm/dashboard/static/built/shared/page-structure.js +251 -0
- claude_mpm/dashboard/static/built/shared/tooltip-service.js +253 -0
- claude_mpm/dashboard/static/built/socket-client.js +1 -1
- claude_mpm/dashboard/static/built/tab-isolation-fix.js +185 -0
- claude_mpm/dashboard/static/css/dashboard.css +28 -5
- claude_mpm/dashboard/static/dist/assets/events.DjpNxWNo.css +1 -0
- claude_mpm/dashboard/static/dist/components/activity-tree.js +1 -1
- claude_mpm/dashboard/static/dist/components/agent-inference.js +1 -1
- claude_mpm/dashboard/static/dist/components/code-viewer.js +2 -0
- claude_mpm/dashboard/static/dist/components/event-processor.js +1 -1
- claude_mpm/dashboard/static/dist/components/event-viewer.js +1 -1
- claude_mpm/dashboard/static/dist/components/export-manager.js +1 -1
- claude_mpm/dashboard/static/dist/components/file-tool-tracker.js +1 -1
- claude_mpm/dashboard/static/dist/components/module-viewer.js +1 -1
- claude_mpm/dashboard/static/dist/components/session-manager.js +1 -1
- claude_mpm/dashboard/static/dist/components/working-directory.js +1 -1
- claude_mpm/dashboard/static/dist/dashboard.js +1 -1
- claude_mpm/dashboard/static/dist/react/events.js +30 -0
- claude_mpm/dashboard/static/dist/socket-client.js +1 -1
- claude_mpm/dashboard/static/events.html +607 -0
- claude_mpm/dashboard/static/index.html +713 -0
- claude_mpm/dashboard/static/js/components/activity-tree.js +3 -17
- claude_mpm/dashboard/static/js/components/agent-hierarchy.js +4 -1
- claude_mpm/dashboard/static/js/components/agent-inference.js +3 -0
- claude_mpm/dashboard/static/js/components/build-tracker.js +8 -0
- claude_mpm/dashboard/static/js/components/code-viewer.js +306 -66
- claude_mpm/dashboard/static/js/components/event-processor.js +3 -0
- claude_mpm/dashboard/static/js/components/event-viewer.js +39 -2
- claude_mpm/dashboard/static/js/components/export-manager.js +3 -0
- claude_mpm/dashboard/static/js/components/file-tool-tracker.js +30 -10
- claude_mpm/dashboard/static/js/components/socket-manager.js +4 -0
- claude_mpm/dashboard/static/js/components/ui-state-manager.js +285 -85
- claude_mpm/dashboard/static/js/components/working-directory.js +3 -0
- claude_mpm/dashboard/static/js/dashboard.js +61 -33
- claude_mpm/dashboard/static/js/socket-client.js +12 -8
- claude_mpm/dashboard/static/js/stores/dashboard-store.js +562 -0
- claude_mpm/dashboard/static/js/tab-isolation-fix.js +185 -0
- claude_mpm/dashboard/static/legacy/activity.html +736 -0
- claude_mpm/dashboard/static/legacy/agents.html +786 -0
- claude_mpm/dashboard/static/legacy/files.html +747 -0
- claude_mpm/dashboard/static/legacy/tools.html +831 -0
- claude_mpm/dashboard/static/monitors-index.html +218 -0
- claude_mpm/dashboard/static/monitors.html +431 -0
- claude_mpm/dashboard/static/production/events.html +659 -0
- claude_mpm/dashboard/static/production/main.html +715 -0
- claude_mpm/dashboard/static/production/monitors.html +483 -0
- claude_mpm/dashboard/static/socket.io.min.js +7 -0
- claude_mpm/dashboard/static/socket.io.v4.8.1.backup.js +7 -0
- claude_mpm/dashboard/static/test-archive/dashboard.html +635 -0
- claude_mpm/dashboard/static/test-archive/debug-events.html +147 -0
- claude_mpm/dashboard/static/test-archive/test-navigation.html +256 -0
- claude_mpm/dashboard/static/test-archive/test-react-exports.html +180 -0
- claude_mpm/dashboard/templates/index.html +79 -9
- claude_mpm/hooks/claude_hooks/services/connection_manager_http.py +1 -1
- claude_mpm/services/agents/deployment/agent_discovery_service.py +3 -0
- claude_mpm/services/agents/deployment/agent_template_builder.py +285 -26
- claude_mpm/services/agents/deployment/agent_validator.py +3 -0
- claude_mpm/services/agents/deployment/validation/template_validator.py +13 -4
- claude_mpm/services/agents/local_template_manager.py +2 -7
- claude_mpm/services/monitor/daemon.py +1 -2
- claude_mpm/services/monitor/daemon_manager.py +2 -7
- claude_mpm/services/monitor/event_emitter.py +6 -2
- claude_mpm/services/monitor/handlers/code_analysis.py +4 -6
- claude_mpm/services/monitor/handlers/hooks.py +2 -6
- claude_mpm/services/monitor/server.py +27 -4
- claude_mpm/tools/code_tree_analyzer.py +2 -4
- claude_mpm/utils/log_cleanup.py +612 -0
- {claude_mpm-4.2.44.dist-info → claude_mpm-4.3.0.dist-info}/METADATA +1 -1
- {claude_mpm-4.2.44.dist-info → claude_mpm-4.3.0.dist-info}/RECORD +151 -83
- claude_mpm/dashboard/static/test-browser-monitor.html +0 -470
- claude_mpm/dashboard/static/test-simple.html +0 -97
- /claude_mpm/dashboard/static/{test_debug.html → test-archive/test_debug.html} +0 -0
- {claude_mpm-4.2.44.dist-info → claude_mpm-4.3.0.dist-info}/WHEEL +0 -0
- {claude_mpm-4.2.44.dist-info → claude_mpm-4.3.0.dist-info}/entry_points.txt +0 -0
- {claude_mpm-4.2.44.dist-info → claude_mpm-4.3.0.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-4.2.44.dist-info → claude_mpm-4.3.0.dist-info}/top_level.txt +0 -0
| @@ -0,0 +1,612 @@ | |
| 1 | 
            +
            """
         | 
| 2 | 
            +
            Comprehensive log cleanup utility for Claude MPM.
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            This module provides automated log cleanup with both size and age-based policies,
         | 
| 5 | 
            +
            including session directory cleanup, archived log removal, and rotation management.
         | 
| 6 | 
            +
            """
         | 
| 7 | 
            +
             | 
| 8 | 
            +
            import gzip
         | 
| 9 | 
            +
            import logging
         | 
| 10 | 
            +
            import os
         | 
| 11 | 
            +
            import shutil
         | 
| 12 | 
            +
            from datetime import datetime, timedelta
         | 
| 13 | 
            +
            from pathlib import Path
         | 
| 14 | 
            +
            from typing import Dict, Optional, Tuple
         | 
| 15 | 
            +
             | 
| 16 | 
            +
            logger = logging.getLogger(__name__)
         | 
| 17 | 
            +
             | 
| 18 | 
            +
             | 
| 19 | 
            +
            class LogCleanupConfig:
         | 
| 20 | 
            +
                """Configuration for log cleanup operations."""
         | 
| 21 | 
            +
             | 
| 22 | 
            +
                # Default retention periods (in days)
         | 
| 23 | 
            +
                DEFAULT_SESSION_MAX_AGE_DAYS = 7
         | 
| 24 | 
            +
                DEFAULT_ARCHIVED_MAX_AGE_DAYS = 30
         | 
| 25 | 
            +
                DEFAULT_LOG_MAX_AGE_DAYS = 14
         | 
| 26 | 
            +
                DEFAULT_PROMPT_MAX_AGE_DAYS = 7
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                # Size thresholds
         | 
| 29 | 
            +
                DEFAULT_MAX_LOG_SIZE_MB = 5
         | 
| 30 | 
            +
                DEFAULT_MAX_TOTAL_SIZE_GB = 1
         | 
| 31 | 
            +
             | 
| 32 | 
            +
                # File patterns
         | 
| 33 | 
            +
                LOG_PATTERNS = {
         | 
| 34 | 
            +
                    "mpm": "mpm_*.log",
         | 
| 35 | 
            +
                    "startup": "startup-*.log",
         | 
| 36 | 
            +
                    "system": "system_*.log",
         | 
| 37 | 
            +
                    "agent": "agent_*.log",
         | 
| 38 | 
            +
                }
         | 
| 39 | 
            +
             | 
| 40 | 
            +
                ARCHIVE_EXTENSIONS = [".gz", ".zip", ".tar", ".bz2"]
         | 
| 41 | 
            +
             | 
| 42 | 
            +
             | 
| 43 | 
            +
            class LogCleanupUtility:
         | 
| 44 | 
            +
                """
         | 
| 45 | 
            +
                Comprehensive log cleanup utility with age and size-based policies.
         | 
| 46 | 
            +
             | 
| 47 | 
            +
                Features:
         | 
| 48 | 
            +
                - Age-based cleanup for session directories
         | 
| 49 | 
            +
                - Cleanup of old archived logs (.gz files)
         | 
| 50 | 
            +
                - Size-based rotation trigger
         | 
| 51 | 
            +
                - Dry-run mode for testing
         | 
| 52 | 
            +
                - Detailed statistics reporting
         | 
| 53 | 
            +
                - Error handling for locked/permission issues
         | 
| 54 | 
            +
                """
         | 
| 55 | 
            +
             | 
| 56 | 
            +
                def __init__(self, base_log_dir: Optional[Path] = None):
         | 
| 57 | 
            +
                    """
         | 
| 58 | 
            +
                    Initialize the log cleanup utility.
         | 
| 59 | 
            +
             | 
| 60 | 
            +
                    Args:
         | 
| 61 | 
            +
                        base_log_dir: Base directory for logs (default: .claude-mpm/logs)
         | 
| 62 | 
            +
                    """
         | 
| 63 | 
            +
                    if base_log_dir is None:
         | 
| 64 | 
            +
                        base_log_dir = Path.cwd() / ".claude-mpm" / "logs"
         | 
| 65 | 
            +
             | 
| 66 | 
            +
                    self.base_log_dir = Path(base_log_dir)
         | 
| 67 | 
            +
                    self.stats = {
         | 
| 68 | 
            +
                        "sessions_removed": 0,
         | 
| 69 | 
            +
                        "archives_removed": 0,
         | 
| 70 | 
            +
                        "logs_removed": 0,
         | 
| 71 | 
            +
                        "space_freed_mb": 0.0,
         | 
| 72 | 
            +
                        "errors": [],
         | 
| 73 | 
            +
                    }
         | 
| 74 | 
            +
             | 
| 75 | 
            +
                def cleanup_old_sessions(
         | 
| 76 | 
            +
                    self,
         | 
| 77 | 
            +
                    max_age_days: int = LogCleanupConfig.DEFAULT_SESSION_MAX_AGE_DAYS,
         | 
| 78 | 
            +
                    dry_run: bool = False,
         | 
| 79 | 
            +
                ) -> Tuple[int, float]:
         | 
| 80 | 
            +
                    """
         | 
| 81 | 
            +
                    Remove session directories older than specified days.
         | 
| 82 | 
            +
             | 
| 83 | 
            +
                    Args:
         | 
| 84 | 
            +
                        max_age_days: Maximum age in days for session directories
         | 
| 85 | 
            +
                        dry_run: If True, only report what would be deleted
         | 
| 86 | 
            +
             | 
| 87 | 
            +
                    Returns:
         | 
| 88 | 
            +
                        Tuple of (directories removed, space freed in MB)
         | 
| 89 | 
            +
                    """
         | 
| 90 | 
            +
                    sessions_dir = self.base_log_dir / "sessions"
         | 
| 91 | 
            +
                    if not sessions_dir.exists():
         | 
| 92 | 
            +
                        logger.info(f"Sessions directory not found: {sessions_dir}")
         | 
| 93 | 
            +
                        return 0, 0.0
         | 
| 94 | 
            +
             | 
| 95 | 
            +
                    cutoff_time = datetime.now() - timedelta(days=max_age_days)
         | 
| 96 | 
            +
                    removed_count = 0
         | 
| 97 | 
            +
                    total_size = 0.0
         | 
| 98 | 
            +
             | 
| 99 | 
            +
                    logger.info(
         | 
| 100 | 
            +
                        f"Scanning for session directories older than {max_age_days} days..."
         | 
| 101 | 
            +
                    )
         | 
| 102 | 
            +
             | 
| 103 | 
            +
                    try:
         | 
| 104 | 
            +
                        for session_dir in sessions_dir.iterdir():
         | 
| 105 | 
            +
                            if not session_dir.is_dir():
         | 
| 106 | 
            +
                                continue
         | 
| 107 | 
            +
             | 
| 108 | 
            +
                            try:
         | 
| 109 | 
            +
                                # Check directory modification time
         | 
| 110 | 
            +
                                mtime = datetime.fromtimestamp(session_dir.stat().st_mtime)
         | 
| 111 | 
            +
             | 
| 112 | 
            +
                                if mtime < cutoff_time:
         | 
| 113 | 
            +
                                    # Calculate directory size
         | 
| 114 | 
            +
                                    dir_size = self._get_directory_size(session_dir)
         | 
| 115 | 
            +
                                    total_size += dir_size
         | 
| 116 | 
            +
             | 
| 117 | 
            +
                                    if dry_run:
         | 
| 118 | 
            +
                                        logger.info(
         | 
| 119 | 
            +
                                            f"[DRY RUN] Would remove session: {session_dir.name} "
         | 
| 120 | 
            +
                                            f"(age: {(datetime.now() - mtime).days} days, "
         | 
| 121 | 
            +
                                            f"size: {dir_size:.2f} MB)"
         | 
| 122 | 
            +
                                        )
         | 
| 123 | 
            +
                                    else:
         | 
| 124 | 
            +
                                        shutil.rmtree(session_dir)
         | 
| 125 | 
            +
                                        logger.info(
         | 
| 126 | 
            +
                                            f"Removed session: {session_dir.name} "
         | 
| 127 | 
            +
                                            f"(age: {(datetime.now() - mtime).days} days, "
         | 
| 128 | 
            +
                                            f"size: {dir_size:.2f} MB)"
         | 
| 129 | 
            +
                                        )
         | 
| 130 | 
            +
             | 
| 131 | 
            +
                                    removed_count += 1
         | 
| 132 | 
            +
             | 
| 133 | 
            +
                            except (PermissionError, OSError) as e:
         | 
| 134 | 
            +
                                error_msg = f"Could not remove {session_dir.name}: {e}"
         | 
| 135 | 
            +
                                logger.warning(error_msg)
         | 
| 136 | 
            +
                                self.stats["errors"].append(error_msg)
         | 
| 137 | 
            +
             | 
| 138 | 
            +
                    except Exception as e:
         | 
| 139 | 
            +
                        logger.error(f"Error scanning sessions directory: {e}")
         | 
| 140 | 
            +
                        self.stats["errors"].append(str(e))
         | 
| 141 | 
            +
             | 
| 142 | 
            +
                    self.stats["sessions_removed"] += removed_count
         | 
| 143 | 
            +
                    self.stats["space_freed_mb"] += total_size
         | 
| 144 | 
            +
             | 
| 145 | 
            +
                    return removed_count, total_size
         | 
| 146 | 
            +
             | 
| 147 | 
            +
                def cleanup_archived_logs(
         | 
| 148 | 
            +
                    self,
         | 
| 149 | 
            +
                    max_age_days: int = LogCleanupConfig.DEFAULT_ARCHIVED_MAX_AGE_DAYS,
         | 
| 150 | 
            +
                    dry_run: bool = False,
         | 
| 151 | 
            +
                ) -> Tuple[int, float]:
         | 
| 152 | 
            +
                    """
         | 
| 153 | 
            +
                    Remove archived log files older than specified days.
         | 
| 154 | 
            +
             | 
| 155 | 
            +
                    Args:
         | 
| 156 | 
            +
                        max_age_days: Maximum age in days for archived files
         | 
| 157 | 
            +
                        dry_run: If True, only report what would be deleted
         | 
| 158 | 
            +
             | 
| 159 | 
            +
                    Returns:
         | 
| 160 | 
            +
                        Tuple of (files removed, space freed in MB)
         | 
| 161 | 
            +
                    """
         | 
| 162 | 
            +
                    cutoff_time = datetime.now() - timedelta(days=max_age_days)
         | 
| 163 | 
            +
                    removed_count = 0
         | 
| 164 | 
            +
                    total_size = 0.0
         | 
| 165 | 
            +
             | 
| 166 | 
            +
                    logger.info(f"Scanning for archived files older than {max_age_days} days...")
         | 
| 167 | 
            +
             | 
| 168 | 
            +
                    # Search for archived files in all subdirectories
         | 
| 169 | 
            +
                    for ext in LogCleanupConfig.ARCHIVE_EXTENSIONS:
         | 
| 170 | 
            +
                        for archive_file in self.base_log_dir.rglob(f"*{ext}"):
         | 
| 171 | 
            +
                            try:
         | 
| 172 | 
            +
                                mtime = datetime.fromtimestamp(archive_file.stat().st_mtime)
         | 
| 173 | 
            +
             | 
| 174 | 
            +
                                if mtime < cutoff_time:
         | 
| 175 | 
            +
                                    file_size = archive_file.stat().st_size / (1024 * 1024)  # MB
         | 
| 176 | 
            +
                                    total_size += file_size
         | 
| 177 | 
            +
             | 
| 178 | 
            +
                                    if dry_run:
         | 
| 179 | 
            +
                                        logger.info(
         | 
| 180 | 
            +
                                            f"[DRY RUN] Would remove archive: {archive_file.name} "
         | 
| 181 | 
            +
                                            f"(age: {(datetime.now() - mtime).days} days, "
         | 
| 182 | 
            +
                                            f"size: {file_size:.2f} MB)"
         | 
| 183 | 
            +
                                        )
         | 
| 184 | 
            +
                                    else:
         | 
| 185 | 
            +
                                        archive_file.unlink()
         | 
| 186 | 
            +
                                        logger.info(
         | 
| 187 | 
            +
                                            f"Removed archive: {archive_file.name} "
         | 
| 188 | 
            +
                                            f"(age: {(datetime.now() - mtime).days} days, "
         | 
| 189 | 
            +
                                            f"size: {file_size:.2f} MB)"
         | 
| 190 | 
            +
                                        )
         | 
| 191 | 
            +
             | 
| 192 | 
            +
                                    removed_count += 1
         | 
| 193 | 
            +
             | 
| 194 | 
            +
                            except (PermissionError, OSError) as e:
         | 
| 195 | 
            +
                                error_msg = f"Could not remove {archive_file.name}: {e}"
         | 
| 196 | 
            +
                                logger.warning(error_msg)
         | 
| 197 | 
            +
                                self.stats["errors"].append(error_msg)
         | 
| 198 | 
            +
             | 
| 199 | 
            +
                    self.stats["archives_removed"] += removed_count
         | 
| 200 | 
            +
                    self.stats["space_freed_mb"] += total_size
         | 
| 201 | 
            +
             | 
| 202 | 
            +
                    return removed_count, total_size
         | 
| 203 | 
            +
             | 
| 204 | 
            +
                def cleanup_old_logs(
         | 
| 205 | 
            +
                    self,
         | 
| 206 | 
            +
                    max_age_days: int = LogCleanupConfig.DEFAULT_LOG_MAX_AGE_DAYS,
         | 
| 207 | 
            +
                    dry_run: bool = False,
         | 
| 208 | 
            +
                    log_type: Optional[str] = None,
         | 
| 209 | 
            +
                ) -> Tuple[int, float]:
         | 
| 210 | 
            +
                    """
         | 
| 211 | 
            +
                    Remove old log files based on age.
         | 
| 212 | 
            +
             | 
| 213 | 
            +
                    Args:
         | 
| 214 | 
            +
                        max_age_days: Maximum age in days for log files
         | 
| 215 | 
            +
                        dry_run: If True, only report what would be deleted
         | 
| 216 | 
            +
                        log_type: Specific log type to clean (mpm, startup, etc.) or None for all
         | 
| 217 | 
            +
             | 
| 218 | 
            +
                    Returns:
         | 
| 219 | 
            +
                        Tuple of (files removed, space freed in MB)
         | 
| 220 | 
            +
                    """
         | 
| 221 | 
            +
                    cutoff_time = datetime.now() - timedelta(days=max_age_days)
         | 
| 222 | 
            +
                    removed_count = 0
         | 
| 223 | 
            +
                    total_size = 0.0
         | 
| 224 | 
            +
             | 
| 225 | 
            +
                    patterns = (
         | 
| 226 | 
            +
                        [LogCleanupConfig.LOG_PATTERNS.get(log_type)]
         | 
| 227 | 
            +
                        if log_type
         | 
| 228 | 
            +
                        else LogCleanupConfig.LOG_PATTERNS.values()
         | 
| 229 | 
            +
                    )
         | 
| 230 | 
            +
             | 
| 231 | 
            +
                    logger.info(f"Scanning for log files older than {max_age_days} days...")
         | 
| 232 | 
            +
             | 
| 233 | 
            +
                    for pattern in patterns:
         | 
| 234 | 
            +
                        for subdir in ["mpm", "startup", "system", "agents"]:
         | 
| 235 | 
            +
                            log_dir = self.base_log_dir / subdir
         | 
| 236 | 
            +
                            if not log_dir.exists():
         | 
| 237 | 
            +
                                continue
         | 
| 238 | 
            +
             | 
| 239 | 
            +
                            for log_file in log_dir.glob(pattern):
         | 
| 240 | 
            +
                                try:
         | 
| 241 | 
            +
                                    mtime = datetime.fromtimestamp(log_file.stat().st_mtime)
         | 
| 242 | 
            +
             | 
| 243 | 
            +
                                    if mtime < cutoff_time:
         | 
| 244 | 
            +
                                        file_size = log_file.stat().st_size / (1024 * 1024)  # MB
         | 
| 245 | 
            +
                                        total_size += file_size
         | 
| 246 | 
            +
             | 
| 247 | 
            +
                                        if dry_run:
         | 
| 248 | 
            +
                                            logger.info(
         | 
| 249 | 
            +
                                                f"[DRY RUN] Would remove log: {log_file.name} "
         | 
| 250 | 
            +
                                                f"(age: {(datetime.now() - mtime).days} days, "
         | 
| 251 | 
            +
                                                f"size: {file_size:.2f} MB)"
         | 
| 252 | 
            +
                                            )
         | 
| 253 | 
            +
                                        else:
         | 
| 254 | 
            +
                                            log_file.unlink()
         | 
| 255 | 
            +
                                            logger.info(
         | 
| 256 | 
            +
                                                f"Removed log: {log_file.name} "
         | 
| 257 | 
            +
                                                f"(age: {(datetime.now() - mtime).days} days, "
         | 
| 258 | 
            +
                                                f"size: {file_size:.2f} MB)"
         | 
| 259 | 
            +
                                            )
         | 
| 260 | 
            +
             | 
| 261 | 
            +
                                        removed_count += 1
         | 
| 262 | 
            +
             | 
| 263 | 
            +
                                except (PermissionError, OSError) as e:
         | 
| 264 | 
            +
                                    error_msg = f"Could not remove {log_file.name}: {e}"
         | 
| 265 | 
            +
                                    logger.warning(error_msg)
         | 
| 266 | 
            +
                                    self.stats["errors"].append(error_msg)
         | 
| 267 | 
            +
             | 
| 268 | 
            +
                    self.stats["logs_removed"] += removed_count
         | 
| 269 | 
            +
                    self.stats["space_freed_mb"] += total_size
         | 
| 270 | 
            +
             | 
| 271 | 
            +
                    return removed_count, total_size
         | 
| 272 | 
            +
             | 
| 273 | 
            +
                def cleanup_empty_directories(self, dry_run: bool = False) -> int:
         | 
| 274 | 
            +
                    """
         | 
| 275 | 
            +
                    Remove empty directories in the log tree.
         | 
| 276 | 
            +
             | 
| 277 | 
            +
                    Args:
         | 
| 278 | 
            +
                        dry_run: If True, only report what would be deleted
         | 
| 279 | 
            +
             | 
| 280 | 
            +
                    Returns:
         | 
| 281 | 
            +
                        Number of directories removed
         | 
| 282 | 
            +
                    """
         | 
| 283 | 
            +
                    removed_count = 0
         | 
| 284 | 
            +
             | 
| 285 | 
            +
                    # Walk bottom-up to remove empty parent directories
         | 
| 286 | 
            +
                    for root, dirs, files in os.walk(self.base_log_dir, topdown=False):
         | 
| 287 | 
            +
                        root_path = Path(root)
         | 
| 288 | 
            +
             | 
| 289 | 
            +
                        # Skip the base log directory itself
         | 
| 290 | 
            +
                        if root_path == self.base_log_dir:
         | 
| 291 | 
            +
                            continue
         | 
| 292 | 
            +
             | 
| 293 | 
            +
                        try:
         | 
| 294 | 
            +
                            # Check if directory is empty
         | 
| 295 | 
            +
                            if not any(root_path.iterdir()):
         | 
| 296 | 
            +
                                if dry_run:
         | 
| 297 | 
            +
                                    logger.info(
         | 
| 298 | 
            +
                                        f"[DRY RUN] Would remove empty directory: {root_path}"
         | 
| 299 | 
            +
                                    )
         | 
| 300 | 
            +
                                else:
         | 
| 301 | 
            +
                                    root_path.rmdir()
         | 
| 302 | 
            +
                                    logger.info(f"Removed empty directory: {root_path}")
         | 
| 303 | 
            +
                                removed_count += 1
         | 
| 304 | 
            +
             | 
| 305 | 
            +
                        except (PermissionError, OSError) as e:
         | 
| 306 | 
            +
                            error_msg = f"Could not remove directory {root_path}: {e}"
         | 
| 307 | 
            +
                            logger.debug(error_msg)  # Debug level since this is common
         | 
| 308 | 
            +
             | 
| 309 | 
            +
                    return removed_count
         | 
| 310 | 
            +
             | 
| 311 | 
            +
                def compress_old_logs(
         | 
| 312 | 
            +
                    self, age_days: int = 7, dry_run: bool = False
         | 
| 313 | 
            +
                ) -> Tuple[int, float]:
         | 
| 314 | 
            +
                    """
         | 
| 315 | 
            +
                    Compress log files older than specified days.
         | 
| 316 | 
            +
             | 
| 317 | 
            +
                    Args:
         | 
| 318 | 
            +
                        age_days: Compress files older than this many days
         | 
| 319 | 
            +
                        dry_run: If True, only report what would be compressed
         | 
| 320 | 
            +
             | 
| 321 | 
            +
                    Returns:
         | 
| 322 | 
            +
                        Tuple of (files compressed, space saved in MB)
         | 
| 323 | 
            +
                    """
         | 
| 324 | 
            +
                    cutoff_time = datetime.now() - timedelta(days=age_days)
         | 
| 325 | 
            +
                    compressed_count = 0
         | 
| 326 | 
            +
                    space_saved = 0.0
         | 
| 327 | 
            +
             | 
| 328 | 
            +
                    for log_file in self.base_log_dir.rglob("*.log"):
         | 
| 329 | 
            +
                        # Skip already compressed files
         | 
| 330 | 
            +
                        if log_file.suffix in LogCleanupConfig.ARCHIVE_EXTENSIONS:
         | 
| 331 | 
            +
                            continue
         | 
| 332 | 
            +
             | 
| 333 | 
            +
                        try:
         | 
| 334 | 
            +
                            mtime = datetime.fromtimestamp(log_file.stat().st_mtime)
         | 
| 335 | 
            +
             | 
| 336 | 
            +
                            if mtime < cutoff_time:
         | 
| 337 | 
            +
                                original_size = log_file.stat().st_size / (1024 * 1024)  # MB
         | 
| 338 | 
            +
                                compressed_path = log_file.with_suffix(".log.gz")
         | 
| 339 | 
            +
             | 
| 340 | 
            +
                                if dry_run:
         | 
| 341 | 
            +
                                    # Estimate compression ratio (typically 80-90% for logs)
         | 
| 342 | 
            +
                                    estimated_saved = original_size * 0.85
         | 
| 343 | 
            +
                                    logger.info(
         | 
| 344 | 
            +
                                        f"[DRY RUN] Would compress: {log_file.name} "
         | 
| 345 | 
            +
                                        f"(size: {original_size:.2f} MB, "
         | 
| 346 | 
            +
                                        f"estimated savings: {estimated_saved:.2f} MB)"
         | 
| 347 | 
            +
                                    )
         | 
| 348 | 
            +
                                    space_saved += estimated_saved
         | 
| 349 | 
            +
                                else:
         | 
| 350 | 
            +
                                    # Actually compress the file
         | 
| 351 | 
            +
                                    with open(log_file, "rb") as f_in:
         | 
| 352 | 
            +
                                        with gzip.open(
         | 
| 353 | 
            +
                                            compressed_path, "wb", compresslevel=9
         | 
| 354 | 
            +
                                        ) as f_out:
         | 
| 355 | 
            +
                                            shutil.copyfileobj(f_in, f_out)
         | 
| 356 | 
            +
             | 
| 357 | 
            +
                                    compressed_size = compressed_path.stat().st_size / (1024 * 1024)
         | 
| 358 | 
            +
                                    saved = original_size - compressed_size
         | 
| 359 | 
            +
                                    space_saved += saved
         | 
| 360 | 
            +
             | 
| 361 | 
            +
                                    # Remove original file
         | 
| 362 | 
            +
                                    log_file.unlink()
         | 
| 363 | 
            +
             | 
| 364 | 
            +
                                    logger.info(
         | 
| 365 | 
            +
                                        f"Compressed: {log_file.name} "
         | 
| 366 | 
            +
                                        f"({original_size:.2f} MB → {compressed_size:.2f} MB, "
         | 
| 367 | 
            +
                                        f"saved: {saved:.2f} MB)"
         | 
| 368 | 
            +
                                    )
         | 
| 369 | 
            +
             | 
| 370 | 
            +
                                compressed_count += 1
         | 
| 371 | 
            +
             | 
| 372 | 
            +
                        except Exception as e:
         | 
| 373 | 
            +
                            error_msg = f"Could not compress {log_file.name}: {e}"
         | 
| 374 | 
            +
                            logger.warning(error_msg)
         | 
| 375 | 
            +
                            self.stats["errors"].append(error_msg)
         | 
| 376 | 
            +
             | 
| 377 | 
            +
                    return compressed_count, space_saved
         | 
| 378 | 
            +
             | 
| 379 | 
            +
                def get_statistics(self) -> Dict:
         | 
| 380 | 
            +
                    """
         | 
| 381 | 
            +
                    Get current statistics about the log directory.
         | 
| 382 | 
            +
             | 
| 383 | 
            +
                    Returns:
         | 
| 384 | 
            +
                        Dictionary with statistics
         | 
| 385 | 
            +
                    """
         | 
| 386 | 
            +
                    stats = {
         | 
| 387 | 
            +
                        "total_size_mb": 0.0,
         | 
| 388 | 
            +
                        "session_count": 0,
         | 
| 389 | 
            +
                        "archive_count": 0,
         | 
| 390 | 
            +
                        "log_count": 0,
         | 
| 391 | 
            +
                        "oldest_session": None,
         | 
| 392 | 
            +
                        "oldest_log": None,
         | 
| 393 | 
            +
                        "directory_sizes": {},
         | 
| 394 | 
            +
                    }
         | 
| 395 | 
            +
             | 
| 396 | 
            +
                    # Calculate total size
         | 
| 397 | 
            +
                    stats["total_size_mb"] = self._get_directory_size(self.base_log_dir)
         | 
| 398 | 
            +
             | 
| 399 | 
            +
                    # Count sessions
         | 
| 400 | 
            +
                    sessions_dir = self.base_log_dir / "sessions"
         | 
| 401 | 
            +
                    if sessions_dir.exists():
         | 
| 402 | 
            +
                        sessions = list(sessions_dir.iterdir())
         | 
| 403 | 
            +
                        stats["session_count"] = len([s for s in sessions if s.is_dir()])
         | 
| 404 | 
            +
             | 
| 405 | 
            +
                        # Find oldest session
         | 
| 406 | 
            +
                        if sessions:
         | 
| 407 | 
            +
                            oldest = min(sessions, key=lambda p: p.stat().st_mtime)
         | 
| 408 | 
            +
                            stats["oldest_session"] = {
         | 
| 409 | 
            +
                                "name": oldest.name,
         | 
| 410 | 
            +
                                "age_days": (
         | 
| 411 | 
            +
                                    datetime.now() - datetime.fromtimestamp(oldest.stat().st_mtime)
         | 
| 412 | 
            +
                                ).days,
         | 
| 413 | 
            +
                            }
         | 
| 414 | 
            +
             | 
| 415 | 
            +
                    # Count archives
         | 
| 416 | 
            +
                    for ext in LogCleanupConfig.ARCHIVE_EXTENSIONS:
         | 
| 417 | 
            +
                        stats["archive_count"] += len(list(self.base_log_dir.rglob(f"*{ext}")))
         | 
| 418 | 
            +
             | 
| 419 | 
            +
                    # Count logs (excluding symlinks)
         | 
| 420 | 
            +
                    stats["log_count"] = len(
         | 
| 421 | 
            +
                        [p for p in self.base_log_dir.rglob("*.log") if not p.is_symlink()]
         | 
| 422 | 
            +
                    )
         | 
| 423 | 
            +
             | 
| 424 | 
            +
                    # Find oldest log (excluding symlinks)
         | 
| 425 | 
            +
                    all_logs = [p for p in self.base_log_dir.rglob("*.log") if not p.is_symlink()]
         | 
| 426 | 
            +
                    if all_logs:
         | 
| 427 | 
            +
                        oldest_log = min(all_logs, key=lambda p: p.stat().st_mtime)
         | 
| 428 | 
            +
                        stats["oldest_log"] = {
         | 
| 429 | 
            +
                            "name": oldest_log.name,
         | 
| 430 | 
            +
                            "path": str(oldest_log.relative_to(self.base_log_dir)),
         | 
| 431 | 
            +
                            "age_days": (
         | 
| 432 | 
            +
                                datetime.now() - datetime.fromtimestamp(oldest_log.stat().st_mtime)
         | 
| 433 | 
            +
                            ).days,
         | 
| 434 | 
            +
                        }
         | 
| 435 | 
            +
             | 
| 436 | 
            +
                    # Calculate directory sizes
         | 
| 437 | 
            +
                    for subdir in ["sessions", "mpm", "startup", "system", "agents", "prompts"]:
         | 
| 438 | 
            +
                        dir_path = self.base_log_dir / subdir
         | 
| 439 | 
            +
                        if dir_path.exists():
         | 
| 440 | 
            +
                            stats["directory_sizes"][subdir] = self._get_directory_size(dir_path)
         | 
| 441 | 
            +
             | 
| 442 | 
            +
                    return stats
         | 
| 443 | 
            +
             | 
| 444 | 
            +
                def perform_full_cleanup(
         | 
| 445 | 
            +
                    self,
         | 
| 446 | 
            +
                    session_max_age_days: int = LogCleanupConfig.DEFAULT_SESSION_MAX_AGE_DAYS,
         | 
| 447 | 
            +
                    archive_max_age_days: int = LogCleanupConfig.DEFAULT_ARCHIVED_MAX_AGE_DAYS,
         | 
| 448 | 
            +
                    log_max_age_days: int = LogCleanupConfig.DEFAULT_LOG_MAX_AGE_DAYS,
         | 
| 449 | 
            +
                    compress_age_days: Optional[int] = None,
         | 
| 450 | 
            +
                    dry_run: bool = False,
         | 
| 451 | 
            +
                ) -> Dict:
         | 
| 452 | 
            +
                    """
         | 
| 453 | 
            +
                    Perform a complete cleanup operation.
         | 
| 454 | 
            +
             | 
| 455 | 
            +
                    Args:
         | 
| 456 | 
            +
                        session_max_age_days: Maximum age for session directories
         | 
| 457 | 
            +
                        archive_max_age_days: Maximum age for archived files
         | 
| 458 | 
            +
                        log_max_age_days: Maximum age for log files
         | 
| 459 | 
            +
                        compress_age_days: Age threshold for compression (None to skip)
         | 
| 460 | 
            +
                        dry_run: If True, only report what would be done
         | 
| 461 | 
            +
             | 
| 462 | 
            +
                    Returns:
         | 
| 463 | 
            +
                        Summary statistics dictionary
         | 
| 464 | 
            +
                    """
         | 
| 465 | 
            +
                    mode = "[DRY RUN] " if dry_run else ""
         | 
| 466 | 
            +
                    logger.info(f"{mode}Starting comprehensive log cleanup...")
         | 
| 467 | 
            +
             | 
| 468 | 
            +
                    # Get initial statistics
         | 
| 469 | 
            +
                    initial_stats = self.get_statistics()
         | 
| 470 | 
            +
             | 
| 471 | 
            +
                    # Reset stats
         | 
| 472 | 
            +
                    self.stats = {
         | 
| 473 | 
            +
                        "sessions_removed": 0,
         | 
| 474 | 
            +
                        "archives_removed": 0,
         | 
| 475 | 
            +
                        "logs_removed": 0,
         | 
| 476 | 
            +
                        "files_compressed": 0,
         | 
| 477 | 
            +
                        "empty_dirs_removed": 0,
         | 
| 478 | 
            +
                        "space_freed_mb": 0.0,
         | 
| 479 | 
            +
                        "space_saved_mb": 0.0,
         | 
| 480 | 
            +
                        "errors": [],
         | 
| 481 | 
            +
                    }
         | 
| 482 | 
            +
             | 
| 483 | 
            +
                    # Cleanup operations
         | 
| 484 | 
            +
                    sessions_removed, sessions_space = self.cleanup_old_sessions(
         | 
| 485 | 
            +
                        session_max_age_days, dry_run
         | 
| 486 | 
            +
                    )
         | 
| 487 | 
            +
             | 
| 488 | 
            +
                    archives_removed, archives_space = self.cleanup_archived_logs(
         | 
| 489 | 
            +
                        archive_max_age_days, dry_run
         | 
| 490 | 
            +
                    )
         | 
| 491 | 
            +
             | 
| 492 | 
            +
                    logs_removed, logs_space = self.cleanup_old_logs(log_max_age_days, dry_run)
         | 
| 493 | 
            +
             | 
| 494 | 
            +
                    # Optional compression
         | 
| 495 | 
            +
                    if compress_age_days is not None:
         | 
| 496 | 
            +
                        compressed, space_saved = self.compress_old_logs(compress_age_days, dry_run)
         | 
| 497 | 
            +
                        self.stats["files_compressed"] = compressed
         | 
| 498 | 
            +
                        self.stats["space_saved_mb"] = space_saved
         | 
| 499 | 
            +
             | 
| 500 | 
            +
                    # Cleanup empty directories
         | 
| 501 | 
            +
                    empty_removed = self.cleanup_empty_directories(dry_run)
         | 
| 502 | 
            +
                    self.stats["empty_dirs_removed"] = empty_removed
         | 
| 503 | 
            +
             | 
| 504 | 
            +
                    # Get final statistics
         | 
| 505 | 
            +
                    final_stats = self.get_statistics() if not dry_run else initial_stats
         | 
| 506 | 
            +
             | 
| 507 | 
            +
                    # Prepare summary
         | 
| 508 | 
            +
                    summary = {
         | 
| 509 | 
            +
                        "mode": "DRY RUN" if dry_run else "EXECUTED",
         | 
| 510 | 
            +
                        "initial_stats": initial_stats,
         | 
| 511 | 
            +
                        "final_stats": final_stats,
         | 
| 512 | 
            +
                        "operations": self.stats,
         | 
| 513 | 
            +
                        "total_removed": (
         | 
| 514 | 
            +
                            self.stats["sessions_removed"]
         | 
| 515 | 
            +
                            + self.stats["archives_removed"]
         | 
| 516 | 
            +
                            + self.stats["logs_removed"]
         | 
| 517 | 
            +
                        ),
         | 
| 518 | 
            +
                        "total_space_impact_mb": (
         | 
| 519 | 
            +
                            self.stats["space_freed_mb"] + self.stats.get("space_saved_mb", 0)
         | 
| 520 | 
            +
                        ),
         | 
| 521 | 
            +
                    }
         | 
| 522 | 
            +
             | 
| 523 | 
            +
                    # Log summary
         | 
| 524 | 
            +
                    logger.info(
         | 
| 525 | 
            +
                        f"{mode}Cleanup complete: "
         | 
| 526 | 
            +
                        f"Removed {summary['total_removed']} items, "
         | 
| 527 | 
            +
                        f"freed {self.stats['space_freed_mb']:.2f} MB"
         | 
| 528 | 
            +
                    )
         | 
| 529 | 
            +
             | 
| 530 | 
            +
                    if self.stats.get("files_compressed"):
         | 
| 531 | 
            +
                        logger.info(
         | 
| 532 | 
            +
                            f"Compressed {self.stats['files_compressed']} files, "
         | 
| 533 | 
            +
                            f"saved {self.stats['space_saved_mb']:.2f} MB"
         | 
| 534 | 
            +
                        )
         | 
| 535 | 
            +
             | 
| 536 | 
            +
                    if self.stats["errors"]:
         | 
| 537 | 
            +
                        logger.warning(
         | 
| 538 | 
            +
                            f"Encountered {len(self.stats['errors'])} errors during cleanup"
         | 
| 539 | 
            +
                        )
         | 
| 540 | 
            +
             | 
| 541 | 
            +
                    return summary
         | 
| 542 | 
            +
             | 
| 543 | 
            +
                def _get_directory_size(self, path: Path) -> float:
         | 
| 544 | 
            +
                    """
         | 
| 545 | 
            +
                    Calculate total size of a directory in MB.
         | 
| 546 | 
            +
             | 
| 547 | 
            +
                    Args:
         | 
| 548 | 
            +
                        path: Directory path
         | 
| 549 | 
            +
             | 
| 550 | 
            +
                    Returns:
         | 
| 551 | 
            +
                        Size in megabytes
         | 
| 552 | 
            +
                    """
         | 
| 553 | 
            +
                    total_size = 0
         | 
| 554 | 
            +
                    try:
         | 
| 555 | 
            +
                        for item in path.rglob("*"):
         | 
| 556 | 
            +
                            if item.is_file():
         | 
| 557 | 
            +
                                total_size += item.stat().st_size
         | 
| 558 | 
            +
                    except Exception as e:
         | 
| 559 | 
            +
                        logger.debug(f"Error calculating size for {path}: {e}")
         | 
| 560 | 
            +
             | 
| 561 | 
            +
                    return total_size / (1024 * 1024)  # Convert to MB
         | 
| 562 | 
            +
             | 
| 563 | 
            +
             | 
| 564 | 
            +
            def run_cleanup_on_startup(
         | 
| 565 | 
            +
                base_log_dir: Optional[Path] = None, config: Optional[Dict] = None
         | 
| 566 | 
            +
            ) -> Optional[Dict]:
         | 
| 567 | 
            +
                """
         | 
| 568 | 
            +
                Run automatic cleanup on application startup.
         | 
| 569 | 
            +
             | 
| 570 | 
            +
                This function is designed to be called during application initialization
         | 
| 571 | 
            +
                to perform routine log maintenance.
         | 
| 572 | 
            +
             | 
| 573 | 
            +
                Args:
         | 
| 574 | 
            +
                    base_log_dir: Base directory for logs
         | 
| 575 | 
            +
                    config: Optional configuration dictionary
         | 
| 576 | 
            +
             | 
| 577 | 
            +
                Returns:
         | 
| 578 | 
            +
                    Cleanup summary or None if disabled
         | 
| 579 | 
            +
                """
         | 
| 580 | 
            +
                # Check if cleanup is enabled
         | 
| 581 | 
            +
                if config and not config.get("auto_cleanup_enabled", True):
         | 
| 582 | 
            +
                    logger.debug("Automatic log cleanup is disabled")
         | 
| 583 | 
            +
                    return None
         | 
| 584 | 
            +
             | 
| 585 | 
            +
                try:
         | 
| 586 | 
            +
                    cleaner = LogCleanupUtility(base_log_dir)
         | 
| 587 | 
            +
             | 
| 588 | 
            +
                    # Use configuration or defaults
         | 
| 589 | 
            +
                    session_days = config.get("session_retention_days", 7) if config else 7
         | 
| 590 | 
            +
                    archive_days = config.get("archive_retention_days", 30) if config else 30
         | 
| 591 | 
            +
                    log_days = config.get("log_retention_days", 14) if config else 14
         | 
| 592 | 
            +
             | 
| 593 | 
            +
                    # Run cleanup (not dry-run)
         | 
| 594 | 
            +
                    summary = cleaner.perform_full_cleanup(
         | 
| 595 | 
            +
                        session_max_age_days=session_days,
         | 
| 596 | 
            +
                        archive_max_age_days=archive_days,
         | 
| 597 | 
            +
                        log_max_age_days=log_days,
         | 
| 598 | 
            +
                        compress_age_days=None,  # Don't compress on startup
         | 
| 599 | 
            +
                        dry_run=False,
         | 
| 600 | 
            +
                    )
         | 
| 601 | 
            +
             | 
| 602 | 
            +
                    logger.info(
         | 
| 603 | 
            +
                        f"Startup cleanup completed: "
         | 
| 604 | 
            +
                        f"Removed {summary['total_removed']} items, "
         | 
| 605 | 
            +
                        f"freed {summary['total_space_impact_mb']:.2f} MB"
         | 
| 606 | 
            +
                    )
         | 
| 607 | 
            +
             | 
| 608 | 
            +
                    return summary
         | 
| 609 | 
            +
             | 
| 610 | 
            +
                except Exception as e:
         | 
| 611 | 
            +
                    logger.error(f"Error during startup cleanup: {e}")
         | 
| 612 | 
            +
                    return None
         |