#!/usr/bin/env python3
"""
Memory Monitoring Utility

This module provides utilities to monitor memory usage during pipeline execution
and help prevent memory leaks and system freezes.
"""

import psutil
import time
import threading
from typing import Dict, Any, Optional
import gc
import os

class MemoryMonitor:
    """Monitor memory usage and provide warnings when thresholds are exceeded"""
    
    def __init__(self, 
                 warning_threshold_mb: int = 4000,  # 4GB warning
                 critical_threshold_mb: int = 6000,  # 6GB critical
                 check_interval: float = 30.0):  # Check every 30 seconds
        self.warning_threshold_mb = warning_threshold_mb
        self.critical_threshold_mb = critical_threshold_mb
        self.check_interval = check_interval
        self.monitoring = False
        self.monitor_thread = None
        self.process = psutil.Process()
        
    def start_monitoring(self):
        """Start background memory monitoring"""
        if self.monitoring:
            return
            
        self.monitoring = True
        self.monitor_thread = threading.Thread(target=self._monitor_loop, daemon=True)
        self.monitor_thread.start()
        print(f"🔍 Memory monitoring started (warning: {self.warning_threshold_mb}MB, critical: {self.critical_threshold_mb}MB)")
    
    def stop_monitoring(self):
        """Stop background memory monitoring"""
        self.monitoring = False
        if self.monitor_thread:
            self.monitor_thread.join(timeout=5.0)
        print("🔍 Memory monitoring stopped")
    
    def _monitor_loop(self):
        """Background monitoring loop"""
        while self.monitoring:
            try:
                memory_info = self.get_memory_usage()
                
                if memory_info['rss_mb'] > self.critical_threshold_mb:
                    print(f"🚨 CRITICAL: Memory usage {memory_info['rss_mb']:.1f}MB exceeds critical threshold!")
                    print(f"   System memory: {memory_info['system_percent']:.1f}% used")
                    self._suggest_cleanup()
                elif memory_info['rss_mb'] > self.warning_threshold_mb:
                    print(f"⚠️  WARNING: Memory usage {memory_info['rss_mb']:.1f}MB exceeds warning threshold")
                    print(f"   System memory: {memory_info['system_percent']:.1f}% used")
                
                time.sleep(self.check_interval)
            except Exception as e:
                print(f"Error in memory monitoring: {e}")
                time.sleep(self.check_interval)
    
    def get_memory_usage(self) -> Dict[str, Any]:
        """Get current memory usage information"""
        process_info = self.process.memory_info()
        system_memory = psutil.virtual_memory()
        
        return {
            'rss_mb': process_info.rss / 1024 / 1024,
            'vms_mb': process_info.vms / 1024 / 1024,
            'percent': self.process.memory_percent(),
            'system_percent': system_memory.percent,
            'system_available_mb': system_memory.available / 1024 / 1024,
            'system_total_mb': system_memory.total / 1024 / 1024
        }
    
    def _suggest_cleanup(self):
        """Suggest cleanup actions when memory is high"""
        print("💡 Suggested actions:")
        print("   1. Force garbage collection: gc.collect()")
        print("   2. Delete large variables: del large_variable")
        print("   3. Restart the process if possible")
        print("   4. Check for memory leaks in the code")
    
    def force_cleanup(self):
        """Force garbage collection and memory cleanup"""
        print("🧹 Forcing memory cleanup...")
        gc.collect()
        memory_info = self.get_memory_usage()
        print(f"   After cleanup: {memory_info['rss_mb']:.1f}MB RSS")
        return memory_info

def get_system_memory_info() -> Dict[str, Any]:
    """Get system-wide memory information"""
    memory = psutil.virtual_memory()
    swap = psutil.swap_memory()
    
    return {
        'total_mb': memory.total / 1024 / 1024,
        'available_mb': memory.available / 1024 / 1024,
        'used_mb': memory.used / 1024 / 1024,
        'percent_used': memory.percent,
        'swap_total_mb': swap.total / 1024 / 1024,
        'swap_used_mb': swap.used / 1024 / 1024,
        'swap_percent': swap.percent
    }

def print_memory_summary():
    """Print a summary of current memory usage"""
    process = psutil.Process()
    process_memory = process.memory_info()
    system_memory = get_system_memory_info()
    
    print("\n" + "="*60)
    print("💾 MEMORY USAGE SUMMARY")
    print("="*60)
    print(f"Process Memory:")
    print(f"  RSS: {process_memory.rss / 1024 / 1024:.1f} MB")
    print(f"  VMS: {process_memory.vms / 1024 / 1024:.1f} MB")
    print(f"  Percent: {process.memory_percent():.1f}%")
    print()
    print(f"System Memory:")
    print(f"  Total: {system_memory['total_mb']:.1f} MB")
    print(f"  Used: {system_memory['used_mb']:.1f} MB ({system_memory['percent_used']:.1f}%)")
    print(f"  Available: {system_memory['available_mb']:.1f} MB")
    print(f"  Swap: {system_memory['swap_used_mb']:.1f} MB / {system_memory['swap_total_mb']:.1f} MB ({system_memory['swap_percent']:.1f}%)")
    print("="*60)

# Context manager for automatic monitoring
class MemoryMonitorContext:
    """Context manager for automatic memory monitoring"""
    
    def __init__(self, **kwargs):
        self.monitor = MemoryMonitor(**kwargs)
    
    def __enter__(self):
        self.monitor.start_monitoring()
        return self.monitor
    
    def __exit__(self, exc_type, exc_val, exc_tb):
        self.monitor.stop_monitoring()

# Example usage:
if __name__ == "__main__":
    print_memory_summary()
    
    # Example of using the context manager
    with MemoryMonitorContext(warning_threshold_mb=1000, critical_threshold_mb=2000) as monitor:
        print("Monitoring memory usage...")
        time.sleep(10)
        monitor.force_cleanup() 