"""
Management command to populate evaluation queues for a new or updated model.

This command creates model_attempts and evaluation_queue entries for active questions
when adding a new model to the system or promoting an existing model to a different tier.

Usage:
    # Add new model to evaluation queue for all active questions (by model name)
    python manage.py populate_model_queue --model grok-4-fast-reasoning

    # Add model by database ID (useful when model_name is not unique)
    python manage.py populate_model_queue --model-id 24

    # Promote model to new tier (creates attempt_number=2 if attempt_number=1 exists)
    python manage.py populate_model_queue --model-id 24 --tier-promotion

    # List what would be created (dry run)
    python manage.py populate_model_queue --model-id 24 --dry-run

    # Target specific questions
    python manage.py populate_model_queue --model-id 24 --questions 14 23 38

Workflow:
1. Finds all active questions (or specified questions)
2. Creates ModelAttempt records for the model
3. Creates EvaluationQueue records with status='cancelled'
4. Supports tier promotion (creates higher attempt numbers for existing models)

Note: If multiple models share the same model_name, the command will display a table
and ask you to use --model-id instead.
"""

from django.core.management.base import BaseCommand
from django.db import transaction
from django.utils import timezone
from questions.models import Question, QuestionState
from model_evaluation.models import Model, ModelAttempt, EvaluationQueue


class Command(BaseCommand):
    help = 'Populate evaluation queues for a model across active questions'

    def add_arguments(self, parser):
        # Model identification (at least one required)
        model_group = parser.add_mutually_exclusive_group(required=True)
        model_group.add_argument(
            '--model',
            type=str,
            help='Model name (e.g., grok-4-fast-reasoning, o4-mini-2025-04-16)'
        )
        model_group.add_argument(
            '--model-id',
            type=int,
            help='Model database ID (use when model_name is ambiguous)'
        )

        parser.add_argument(
            '--tier-promotion',
            action='store_true',
            help='Handle tier promotion (creates attempt_number=2 if attempt_number=1 exists)'
        )
        parser.add_argument(
            '--dry-run',
            action='store_true',
            help='Show what would be created without actually creating records'
        )
        parser.add_argument(
            '--questions',
            nargs='*',
            type=int,
            help='Specific question IDs to populate (default: all active questions)'
        )

    def handle(self, *args, **options):
        model_name = options.get('model')
        model_id = options.get('model_id')
        tier_promotion = options['tier_promotion']
        dry_run = options['dry_run']
        specific_questions = options['questions']

        # Get the model
        model = None

        if model_id:
            # Direct lookup by ID
            try:
                model = Model.objects.get(id=model_id)
                self.stdout.write(self.style.SUCCESS(f'✅ Found model by ID: {model.display_name} (ID: {model_id})'))
            except Model.DoesNotExist:
                self.stdout.write(self.style.ERROR(f'❌ Model with ID {model_id} not found in database'))
                self.stdout.write('\nAvailable models:')
                for m in Model.objects.filter(is_active=True).order_by('tier__tier_number', 'model_name'):
                    self.stdout.write(f'  ID {m.id}: {m.model_name} ({m.display_name}) - {m.framework_type} - Tier {m.tier.tier_number}')
                return

        else:
            # Lookup by model name
            try:
                model = Model.objects.get(model_name=model_name)
            except Model.DoesNotExist:
                self.stdout.write(self.style.ERROR(f'❌ Model "{model_name}" not found in database'))
                self.stdout.write('\nAvailable models:')
                for m in Model.objects.filter(is_active=True).order_by('tier__tier_number', 'model_name'):
                    self.stdout.write(f'  ID {m.id}: {m.model_name} ({m.display_name}) - {m.framework_type} - Tier {m.tier.tier_number}')
                return
            except Model.MultipleObjectsReturned:
                # Multiple models with same name - show table and ask for confirmation
                matching_models = Model.objects.filter(model_name=model_name).order_by('id')

                self.stdout.write(self.style.WARNING(f'\n⚠️  Multiple models found with model_name "{model_name}":'))
                self.stdout.write('\n{:<6} {:<30} {:<35} {:<25}'.format('ID', 'Model Name', 'Display Name', 'Framework Type'))
                self.stdout.write('=' * 100)
                for m in matching_models:
                    self.stdout.write('{:<6} {:<30} {:<35} {:<25}'.format(
                        m.id,
                        m.model_name,
                        m.display_name or '',
                        m.framework_type or ''
                    ))

                self.stdout.write('\n💡 Please specify which model you want using --model-id:')
                for m in matching_models:
                    self.stdout.write(f'   python manage.py populate_model_queue --model-id {m.id} [other options]')
                return

        self.stdout.write(self.style.SUCCESS(f'\n📋 Populating queue for: {model.display_name}'))
        self.stdout.write(f'   Model name: {model.model_name}')
        self.stdout.write(f'   Framework: {model.framework_type}')
        self.stdout.write(f'   Tier: {model.tier.tier_number} - {model.tier.tier_name}')
        self.stdout.write(f'   Company: {model.company.company_name}')

        # Get questions to process
        if specific_questions:
            questions = Question.objects.filter(id__in=specific_questions)
            self.stdout.write(f'\n🎯 Target: {len(specific_questions)} specified questions')
        else:
            active_state = QuestionState.objects.get(status='active')
            questions = Question.objects.filter(status=active_state)
            self.stdout.write(f'\n🎯 Target: All active questions ({questions.count()} found)')

        if not questions.exists():
            self.stdout.write(self.style.WARNING('⚠️  No questions to process'))
            return

        # Process each question
        created_attempts = 0
        skipped_attempts = 0
        created_queue_entries = 0

        with transaction.atomic():
            for question in questions:
                # Determine attempt number
                existing_attempts = ModelAttempt.objects.filter(
                    model=model,
                    question=question
                ).order_by('-attempt_number')

                if existing_attempts.exists():
                    max_attempt = existing_attempts.first().attempt_number

                    if tier_promotion:
                        # Create new attempt with incremented number
                        attempt_number = max_attempt + 1
                        self.stdout.write(f'  Q{question.id}: Creating attempt #{attempt_number} (tier promotion)')
                    else:
                        # Skip - attempt already exists
                        self.stdout.write(self.style.WARNING(
                            f'  Q{question.id}: Skipped (attempt #{max_attempt} already exists)'
                        ))
                        skipped_attempts += 1
                        continue
                else:
                    # First attempt for this model-question pair
                    attempt_number = 1
                    self.stdout.write(f'  Q{question.id}: Creating attempt #{attempt_number} (new)')

                if dry_run:
                    self.stdout.write(f'    [DRY RUN] Would create: ModelAttempt(attempt_number={attempt_number})')
                    self.stdout.write(f'    [DRY RUN] Would create: EvaluationQueue(status=cancelled)')
                    created_attempts += 1
                    created_queue_entries += 1
                else:
                    # Create ModelAttempt with explicit time (unmanaged model doesn't auto-populate)
                    attempt = ModelAttempt.objects.create(
                        model=model,
                        question=question,
                        attempt_number=attempt_number,
                        time=timezone.now()
                    )
                    created_attempts += 1

                    # Create EvaluationQueue entry with status='cancelled'
                    queue_entry = EvaluationQueue.objects.create(
                        attempt=attempt,
                        status='cancelled'
                    )
                    created_queue_entries += 1

                    self.stdout.write(f'    ✅ Created: Attempt #{attempt_number}, Queue entry (cancelled)')

        # Summary
        self.stdout.write('\n' + '='*60)
        if dry_run:
            self.stdout.write(self.style.WARNING('🔍 DRY RUN - No changes made'))
        else:
            self.stdout.write(self.style.SUCCESS('✅ Queue population complete'))

        self.stdout.write(f'\n📊 Summary:')
        self.stdout.write(f'   Questions processed: {questions.count()}')
        self.stdout.write(f'   Attempts created: {created_attempts}')
        self.stdout.write(f'   Attempts skipped: {skipped_attempts}')
        self.stdout.write(f'   Queue entries created: {created_queue_entries}')

        if created_queue_entries > 0:
            self.stdout.write(f'\n💡 Next steps:')
            self.stdout.write(f'   1. Review queue: python manage.py manage_evaluation_queue --status')
            self.stdout.write(f'   2. Test single evaluation: Use web UI to re-run one cancelled evaluation')
            self.stdout.write(f'   3. Bulk re-run: Use "Select All Cancelled" + "Re-run Selected" in web UI')
            self.stdout.write(f'   4. Monitor: python manage.py run_evaluation_queue')
