"""
Django Management Command for Evaluation Queue Operations

This command provides utilities for managing the evaluation queue,
including adding evaluations, monitoring status, and cleanup operations.
"""

from django.core.management.base import BaseCommand, CommandError
from django.db import transaction
from django.utils import timezone
from django.db.models import Q

from model_evaluation.models import (
    Model, Question, EvaluationQueue, ModelTier, Company
)
from model_evaluation.queue_manager import queue_manager, get_queue_stats
import json


class Command(BaseCommand):
    help = 'Manage evaluation queue operations'

    def add_arguments(self, parser):
        """Add command line arguments."""
        subparsers = parser.add_subparsers(dest='action', help='Available actions')
        
        # Status command
        status_parser = subparsers.add_parser('status', help='Show queue status')
        status_parser.add_argument(
            '--detailed', 
            action='store_true', 
            help='Show detailed status with individual items'
        )
        
        # Add command
        add_parser = subparsers.add_parser('add', help='Add evaluations to queue')
        add_parser.add_argument('--model', type=str, help='Model name to evaluate')
        add_parser.add_argument('--question', type=int, help='Question ID to evaluate')
        add_parser.add_argument('--tier', type=int, help='Add all models from specific tier')
        add_parser.add_argument('--all-active', action='store_true', 
                               help='Add all active questions for all active models')
        
        # Cleanup command
        cleanup_parser = subparsers.add_parser('cleanup', help='Cleanup operations')
        cleanup_parser.add_argument(
            '--stale-locks', 
            action='store_true', 
            help='Clean up stale company locks'
        )
        cleanup_parser.add_argument(
            '--failed-items', 
            action='store_true', 
            help='Clean up old failed queue items'
        )
        cleanup_parser.add_argument(
            '--max-age-hours', 
            type=int, 
            default=24,
            help='Maximum age in hours for cleanup operations'
        )
        
        # Monitor command
        monitor_parser = subparsers.add_parser('monitor', help='Monitor queue activity')
        monitor_parser.add_argument(
            '--interval', 
            type=int, 
            default=30,
            help='Refresh interval in seconds'
        )
        
        # Reset command
        reset_parser = subparsers.add_parser('reset', help='Reset queue items')
        reset_parser.add_argument(
            '--pending-to-failed', 
            action='store_true',
            help='Mark all pending items as failed'
        )
        reset_parser.add_argument(
            '--running-to-failed', 
            action='store_true',
            help='Mark all running items as failed'
        )

    def handle(self, *args, **options):
        """Handle the command execution."""
        action = options.get('action')
        
        if not action:
            self.print_help('manage.py', 'manage_evaluation_queue')
            return
        
        try:
            if action == 'status':
                self.handle_status(options)
            elif action == 'add':
                self.handle_add(options)
            elif action == 'cleanup':
                self.handle_cleanup(options)
            elif action == 'monitor':
                self.handle_monitor(options)
            elif action == 'reset':
                self.handle_reset(options)
            else:
                raise CommandError(f"Unknown action: {action}")
                
        except Exception as e:
            raise CommandError(f"Command failed: {str(e)}")

    def handle_status(self, options):
        """Handle status command."""
        self.stdout.write(self.style.SUCCESS("Evaluation Queue Status"))
        self.stdout.write("=" * 50)
        
        # Get basic statistics
        stats = get_queue_stats()
        
        # Display summary
        self.stdout.write(f"Pending:   {stats.get('pending', 0)}")
        self.stdout.write(f"Running:   {stats.get('running', 0)}")
        self.stdout.write(f"Completed: {stats.get('completed', 0)}")
        self.stdout.write(f"Failed:    {stats.get('failed', 0)}")
        
        # Display tier breakdown
        if 'by_tier' in stats:
            self.stdout.write("\nBy Tier:")
            for tier, count in stats['by_tier'].items():
                tier_num = tier.replace('tier_', '')
                self.stdout.write(f"  Tier {tier_num}: {count} pending")
        
        # Display company locks
        if 'company_locks' in stats:
            self.stdout.write("\nCompany Locks:")
            for company, lock_info in stats['company_locks'].items():
                status = "🔒 LOCKED" if lock_info['is_locked'] else "🔓 Available"
                self.stdout.write(f"  {company}: {status}")
                if lock_info['is_locked'] and lock_info['locked_at']:
                    self.stdout.write(f"    Locked since: {lock_info['locked_at']}")
        
        # Show detailed items if requested
        if options.get('detailed'):
            self.stdout.write("\nDetailed Queue Items:")
            self._show_detailed_queue()

    def _show_detailed_queue(self):
        """Show detailed queue items."""
        queue_items = EvaluationQueue.objects.select_related(
            'attempt__model__company', 'attempt__model__tier', 'attempt__question'
        ).order_by('-submitted_at')[:20]  # Show newest first, limit to 20 items
        
        if not queue_items:
            self.stdout.write("  No items in queue")
            return
        
        for item in queue_items:
            status_color = {
                'pending': self.style.WARNING,
                'running': self.style.NOTICE,
                'completed': self.style.SUCCESS,
                'failed': self.style.ERROR,
            }.get(item.status, self.style.NOTICE)
            
            status_text = item.status.upper().ljust(8)
            model_name = item.attempt.model.model_name[:20].ljust(20)
            question_title = item.attempt.question.title[:30].ljust(30)
            
            self.stdout.write(
                f"  [{item.id:4d}] {status_color(status_text)} "
                f"T{item.attempt.model.tier.tier_number} {model_name} "
                f"Q{item.attempt.question.id:3d} {question_title} "
                f"(attempt {item.attempt.attempt_number})"
            )

    def handle_add(self, options):
        """Handle add command."""
        added_count = 0
        
        if options.get('all_active'):
            # Add all active questions for all active models
            active_models = Model.objects.filter(is_active=True)
            active_questions = Question.objects.filter(status=9)
            
            self.stdout.write(f"Adding {active_models.count()} models × {active_questions.count()} questions...")
            
            pairs = [(model, question) for model in active_models for question in active_questions]
            created_items = queue_manager.bulk_add_evaluations(pairs)
            added_count = len(created_items)
            
        elif options.get('tier'):
            # Add all models from specific tier for all active questions
            tier_num = options['tier']
            tier_models = Model.objects.filter(tier__tier_number=tier_num, is_active=True)
            active_questions = Question.objects.filter(status=9)
            
            self.stdout.write(f"Adding Tier {tier_num} models × active questions...")
            
            pairs = [(model, question) for model in tier_models for question in active_questions]
            created_items = queue_manager.bulk_add_evaluations(pairs)
            added_count = len(created_items)
            
        elif options.get('model') and options.get('question'):
            # Add specific model and question
            try:
                model = Model.objects.get(model_name=options['model'])
                question = Question.objects.get(id=options['question'])
                
                queue_item = queue_manager.add_to_queue(model, question)
                added_count = 1
                
                self.stdout.write(f"Added: {model.model_name} × Q{question.id}")
                
            except Model.DoesNotExist:
                raise CommandError(f"Model not found: {options['model']}")
            except Question.DoesNotExist:
                raise CommandError(f"Question not found: {options['question']}")
                
        else:
            raise CommandError("Specify --all-active, --tier N, or --model NAME --question ID")
        
        self.stdout.write(
            self.style.SUCCESS(f"Successfully added {added_count} evaluations to queue")
        )

    def handle_cleanup(self, options):
        """Handle cleanup command."""
        max_age_hours = options.get('max_age_hours', 24)
        cleaned_count = 0
        
        if options.get('stale_locks'):
            # Clean up stale execution trackers
            cleaned_count += queue_manager.cleanup_stale_trackers(max_age_hours)
            self.stdout.write(f"Cleaned up {cleaned_count} stale execution trackers")
        
        if options.get('failed_items'):
            # Clean up old failed items
            cutoff_time = timezone.now() - timezone.timedelta(hours=max_age_hours)
            
            with transaction.atomic():
                old_failed = EvaluationQueue.objects.filter(
                    status='failed',
                    completed_at__lt=cutoff_time
                )
                count = old_failed.count()
                old_failed.delete()
                cleaned_count += count
                
            self.stdout.write(f"Cleaned up {count} old failed queue items")
        
        if cleaned_count == 0:
            self.stdout.write("Nothing to clean up")
        else:
            self.stdout.write(
                self.style.SUCCESS(f"Total cleanup: {cleaned_count} items")
            )

    def handle_monitor(self, options):
        """Handle monitor command (simplified version)."""
        import time
        
        interval = options.get('interval', 30)
        self.stdout.write(f"Monitoring queue (refresh every {interval}s, Ctrl+C to stop)...")
        
        try:
            while True:
                # Clear screen (simple version)
                self.stdout.write("\n" + "=" * 60)
                self.stdout.write(f"Queue Status at {timezone.now().strftime('%Y-%m-%d %H:%M:%S')}")
                self.stdout.write("=" * 60)
                
                # Show status
                self.handle_status({'detailed': False})
                
                time.sleep(interval)
                
        except KeyboardInterrupt:
            self.stdout.write("\nMonitoring stopped")

    def handle_reset(self, options):
        """Handle reset command."""
        reset_count = 0
        
        if options.get('pending_to_failed'):
            with transaction.atomic():
                pending_items = EvaluationQueue.objects.filter(status='pending')
                count = pending_items.update(
                    status='failed',
                    error_message='Reset by admin command',
                    completed_at=timezone.now()
                )
                reset_count += count
                
            self.stdout.write(f"Reset {count} pending items to failed")
        
        if options.get('running_to_failed'):
            with transaction.atomic():
                running_items = EvaluationQueue.objects.filter(status='running')
                count = running_items.update(
                    status='failed',
                    error_message='Reset by admin command (was running)',
                    completed_at=timezone.now()
                )
                reset_count += count
                
                # Also release execution trackers
                queue_manager.cleanup_stale_trackers(max_age_hours=0)
                
            self.stdout.write(f"Reset {count} running items to failed and released locks")
        
        if reset_count > 0:
            self.stdout.write(
                self.style.WARNING(f"Total reset: {reset_count} items")
            )
        else:
            self.stdout.write("No items to reset")