"""
Django Management Command to Clean Up Stale Execution Trackers

This command removes stale execution trackers and fixes inconsistent states
that may occur from cancelled evaluations or system crashes.
"""

from django.core.management.base import BaseCommand
from django.utils import timezone
from datetime import timedelta

from model_evaluation.models import ExecutionTracker, EvaluationQueue, CompanyExecutionLock
from model_evaluation.queue_manager import queue_manager


class Command(BaseCommand):
    help = 'Clean up stale execution trackers and locks'

    def add_arguments(self, parser):
        parser.add_argument(
            '--max-age',
            type=int,
            default=3,
            help='Maximum age of trackers in hours before considering them stale (default: 3)'
        )
        parser.add_argument(
            '--dry-run',
            action='store_true',
            help='Show what would be cleaned up without actually doing it'
        )
        parser.add_argument(
            '--clean-old-locks',
            action='store_true',
            help='Also clean up old CompanyExecutionLock entries'
        )

    def handle(self, *args, **options):
        max_age = options['max_age']
        dry_run = options['dry_run']
        clean_old_locks = options['clean_old_locks']
        
        cutoff_time = timezone.now() - timedelta(hours=max_age)
        
        self.stdout.write(f"Cleaning up trackers older than {max_age} hours...")
        self.stdout.write(f"Cutoff time: {cutoff_time}")
        
        if dry_run:
            self.stdout.write(self.style.WARNING("DRY RUN - No changes will be made"))
        
        # Find stale execution trackers
        stale_trackers = ExecutionTracker.objects.filter(
            started_at__lt=cutoff_time
        ).select_related('queue', 'company', 'model')
        
        self.stdout.write(f"\nFound {stale_trackers.count()} stale execution trackers:")
        
        for tracker in stale_trackers:
            self.stdout.write(
                f"  - Queue {tracker.queue_id}: {tracker.model.model_name} on "
                f"Q{tracker.question_id} (Company: {tracker.company.company_name}, "
                f"Started: {tracker.started_at}, PID: {tracker.subprocess_pid})"
            )
            
            # Update queue status if still running
            if tracker.queue and tracker.queue.status == 'running':
                self.stdout.write(f"    └─ Queue item still marked as running, will mark as failed")
                if not dry_run:
                    tracker.queue.status = 'failed'
                    tracker.queue.error_message = f'Evaluation timed out (stale tracker cleanup after {max_age} hours)'
                    tracker.queue.completed_at = timezone.now()
                    tracker.queue.save()
        
        if not dry_run and stale_trackers.count() > 0:
            deleted_count = stale_trackers.delete()[0]
            self.stdout.write(
                self.style.SUCCESS(f"Deleted {deleted_count} stale execution trackers")
            )
        
        # Clean up old CompanyExecutionLock entries if requested
        if clean_old_locks:
            self.stdout.write("\nChecking for old CompanyExecutionLock entries...")
            try:
                old_locks = CompanyExecutionLock.objects.filter(
                    locked_at__lt=cutoff_time
                ).select_related('company')
                
                self.stdout.write(f"Found {old_locks.count()} old company locks:")
                
                for lock in old_locks:
                    self.stdout.write(
                        f"  - {lock.company.company_name} locked at {lock.locked_at} "
                        f"by queue {lock.locked_by_queue_id}"
                    )
                
                if not dry_run and old_locks.count() > 0:
                    deleted_count = old_locks.delete()[0]
                    self.stdout.write(
                        self.style.SUCCESS(f"Deleted {deleted_count} old company locks")
                    )
            except Exception as e:
                self.stdout.write(
                    self.style.WARNING(f"Could not clean old locks (table may not exist): {e}")
                )
        
        # Check for inconsistencies
        self.stdout.write("\nChecking for inconsistencies...")
        
        # Find running queue items without execution trackers
        running_without_tracker = EvaluationQueue.objects.filter(
            status='running'
        ).exclude(
            id__in=ExecutionTracker.objects.values_list('queue_id', flat=True)
        )
        
        if running_without_tracker.count() > 0:
            self.stdout.write(
                self.style.WARNING(
                    f"Found {running_without_tracker.count()} running evaluations without trackers:"
                )
            )
            for queue_item in running_without_tracker:
                self.stdout.write(f"  - Queue {queue_item.id}: Started at {queue_item.started_at}")
                if not dry_run:
                    # Mark as failed since there's no tracker
                    queue_item.status = 'failed'
                    queue_item.error_message = 'No execution tracker found (inconsistent state)'
                    queue_item.completed_at = timezone.now()
                    queue_item.save()
            
            if not dry_run:
                self.stdout.write(
                    self.style.SUCCESS(
                        f"Fixed {running_without_tracker.count()} inconsistent queue items"
                    )
                )
        else:
            self.stdout.write(self.style.SUCCESS("No inconsistencies found"))
        
        # Summary
        self.stdout.write("\n" + "="*50)
        if dry_run:
            self.stdout.write("Dry run complete - no changes were made")
        else:
            self.stdout.write("Cleanup complete")
            
            # Use queue manager's cleanup method as well
            additional_cleaned = queue_manager.cleanup_stale_trackers(max_age_hours=max_age)
            if additional_cleaned > 0:
                self.stdout.write(
                    f"Queue manager cleaned up {additional_cleaned} additional items"
                )