"""
Django Management Command to Run the Evaluation Queue Service

This command starts the evaluation executor service that processes
the evaluation queue and runs mathematical model evaluations.
"""

import asyncio
import signal
import sys
import os
from pathlib import Path
from django.core.management.base import BaseCommand, CommandError
from django.conf import settings

# Load environment variables from .env file
from dotenv import load_dotenv
env_path = Path(__file__).parent.parent.parent.parent / '.env'
if env_path.exists():
    load_dotenv(dotenv_path=env_path)

from model_evaluation.evaluation_executor import evaluation_executor, EvaluationRunner
from model_evaluation.queue_manager import get_queue_stats
from model_evaluation.db_config import validate_environment


class Command(BaseCommand):
    help = 'Run the evaluation queue service'

    def add_arguments(self, parser):
        """Add command line arguments."""
        parser.add_argument(
            '--single',
            action='store_true',
            help='Run a single evaluation instead of the service'
        )
        parser.add_argument(
            '--model',
            type=str,
            help='Model name for single evaluation'
        )
        parser.add_argument(
            '--question',
            type=int,
            help='Question ID for single evaluation'
        )
        parser.add_argument(
            '--attempt',
            type=int,
            default=1,
            help='Attempt number for single evaluation (1 or 2)'
        )
        parser.add_argument(
            '--dry-run',
            action='store_true',
            help='Check configuration without starting service'
        )
        parser.add_argument(
            '--status',
            action='store_true',
            help='Show service and queue status'
        )

    def handle(self, *args, **options):
        """Handle the command execution."""
        try:
            if options.get('status'):
                self.handle_status()
                return
            
            if options.get('dry_run'):
                self.handle_dry_run()
                return
            
            if options.get('single'):
                self.handle_single_evaluation(options)
                return
            
            # Start the main service
            self.handle_service()
            
        except KeyboardInterrupt:
            self.stdout.write("\nShutdown requested by user")
        except Exception as e:
            raise CommandError(f"Command failed: {str(e)}")

    def handle_status(self):
        """Handle status command."""
        self.stdout.write(self.style.SUCCESS("Evaluation Service Status"))
        self.stdout.write("=" * 50)
        
        # Service status
        executor_status = evaluation_executor.get_service_status()
        
        self.stdout.write(f"Service Running: {executor_status['is_running']}")
        self.stdout.write(f"Current Evaluations: {executor_status['current_evaluations']}")
        # Note: No longer limited by max_concurrent - parallel evaluation by company
        self.stdout.write("Token-based evaluation limits enabled")
        
        # Queue status
        self.stdout.write("\nQueue Status:")
        queue_stats = get_queue_stats()
        
        self.stdout.write(f"  Pending: {queue_stats.get('pending', 0)}")
        self.stdout.write(f"  Running: {queue_stats.get('running', 0)}")
        self.stdout.write(f"  Completed: {queue_stats.get('completed', 0)}")
        self.stdout.write(f"  Failed: {queue_stats.get('failed', 0)}")
        
        # Running evaluations
        if executor_status['running_evaluations']:
            self.stdout.write("\nCurrently Running Evaluations:")
            for eval_info in executor_status['running_evaluations']:
                duration = int(eval_info['duration_seconds'])
                self.stdout.write(
                    f"  Queue {eval_info['queue_id']}: {eval_info['model']} "
                    f"on Q{eval_info['question_id']} ({duration}s)"
                )

    def handle_dry_run(self):
        """Handle dry run command."""
        self.stdout.write("Evaluation Service Configuration Check")
        self.stdout.write("=" * 50)
        
        # Check environment
        self.stdout.write("Checking API keys...")
        env_valid = validate_environment()
        
        if env_valid:
            self.stdout.write(self.style.SUCCESS("✓ All API keys present"))
        else:
            self.stdout.write(self.style.WARNING("⚠ Some API keys missing"))
        
        # Check configuration
        self.stdout.write("Checking configuration...")
        
        timeout = getattr(settings, 'EVALUATION_TIMEOUT', 86400)
        poll_interval = getattr(settings, 'QUEUE_POLL_INTERVAL', 30)
        
        self.stdout.write(f"  Evaluation timeout: {timeout}s ({timeout/3600:.1f}h)")
        self.stdout.write(f"  Parallel evaluation: Enabled (company-based)")
        self.stdout.write(f"  Poll interval: {poll_interval}s")
        
        # Check queue
        self.stdout.write("Checking queue...")
        queue_stats = get_queue_stats()
        pending = queue_stats.get('pending', 0)
        
        if pending > 0:
            self.stdout.write(f"  {pending} evaluations pending")
        else:
            self.stdout.write("  No evaluations pending")
        
        self.stdout.write(self.style.SUCCESS("Configuration check complete"))

    def handle_single_evaluation(self, options):
        """Handle single evaluation command."""
        model = options.get('model')
        question = options.get('question')
        attempt = options.get('attempt', 1)
        
        if not model or not question:
            raise CommandError("Single evaluation requires --model and --question")
        
        self.stdout.write(f"Running single evaluation: {model} on Q{question} (attempt {attempt})")
        
        # Run async single evaluation
        asyncio.run(self._run_single_evaluation(model, question, attempt))

    async def _run_single_evaluation(self, model: str, question_id: int, attempt: int):
        """Run a single evaluation asynchronously."""
        runner = EvaluationRunner()
        
        try:
            result = await runner.run_single_evaluation(model, question_id, attempt)
            
            # Display results
            self.stdout.write("\nEvaluation Results:")
            self.stdout.write("=" * 30)
            
            success = result.get('success', False)
            if success:
                self.stdout.write(self.style.SUCCESS(f"Status: SUCCESS"))
                
                if 'model_output' in result:
                    output = result['model_output'][:200] + "..." if len(result['model_output']) > 200 else result['model_output']
                    self.stdout.write(f"Output: {output}")
                
                if 'score' in result:
                    self.stdout.write(f"Score: {result['score']}")
                
                if 'duration' in result:
                    self.stdout.write(f"Duration: {result['duration']:.1f}s")
            else:
                self.stdout.write(self.style.ERROR(f"Status: FAILED"))
                if 'error' in result:
                    self.stdout.write(f"Error: {result['error']}")
            
        except Exception as e:
            self.stdout.write(self.style.ERROR(f"Single evaluation failed: {str(e)}"))

    def handle_service(self):
        """Handle main service command."""
        self.stdout.write("Starting Evaluation Queue Service")
        self.stdout.write("=" * 40)
        
        # Pre-flight checks
        self.stdout.write("Running pre-flight checks...")
        
        # Check environment
        if not validate_environment():
            self.stdout.write(self.style.WARNING("Some API keys are missing. Some evaluations may fail."))
        
        # Ensure API key environment variables are available for subprocess calls
        # Read from database synchronously (main process context)
        try:
            from model_evaluation.models import Company
            for company in Company.objects.all():
                if company.api_key and company.api_key in os.environ:
                    # Ensure it's available for subprocess calls
                    os.environ[company.api_key] = os.environ[company.api_key]
        except Exception as env_error:
            self.stdout.write(f"⚠️  Warning: Could not setup environment: {env_error}")
        
        # Check queue
        queue_stats = get_queue_stats()
        pending = queue_stats.get('pending', 0)
        
        if pending == 0:
            self.stdout.write("No evaluations pending. Service will wait for new work.")
        else:
            self.stdout.write(f"Found {pending} pending evaluations")
        
        self.stdout.write("Pre-flight checks complete")
        self.stdout.write("=" * 40)
        
        # Display service configuration
        executor_status = evaluation_executor.get_service_status()
        self.stdout.write("Service Configuration:")
        self.stdout.write(f"  Parallel evaluation: Enabled by company (unlimited)")
        self.stdout.write(f"  Token limits: Main=300k, Subquestion=100k, Max per call=100k")
        self.stdout.write(f"  Queue poll interval: {executor_status['poll_interval']}s")
        
        self.stdout.write("\nStarting service... (Press Ctrl+C to stop)")
        
        # Setup signal handling for graceful shutdown
        def signal_handler(signum, frame):
            self.stdout.write(f"\nReceived signal {signum}, initiating graceful shutdown...")
            # The executor will handle the shutdown
        
        signal.signal(signal.SIGINT, signal_handler)
        signal.signal(signal.SIGTERM, signal_handler)
        
        # Start the service
        try:
            asyncio.run(evaluation_executor.start_service())
        except KeyboardInterrupt:
            pass  # Handled by signal handler
        except Exception as e:
            self.stdout.write(self.style.ERROR(f"Service error: {str(e)}"))
            raise CommandError(f"Service failed: {str(e)}")
        
        self.stdout.write("Service stopped")