"""
Views for AI testing functionality.

Provides AJAX endpoints for testing questions with OpenAI API.
"""

import json
import logging
from django.http import JsonResponse
from django.views.decorators.http import require_POST, require_GET
from django.contrib.auth.decorators import login_required
from django.shortcuts import get_object_or_404
from django_q.tasks import async_task, fetch

from questions.models import Question
from .models import TestAttempt
from .services import OpenAITestService, async_test_question


logger = logging.getLogger(__name__)


@require_POST
@login_required
def test_question_ajax(request, question_id):
    """
    AJAX endpoint for initiating async testing of a question with OpenAI API.
    
    This endpoint creates a TestAttempt record and queues an async task,
    returning immediately with a task_id for status polling.
    
    Expected POST data:
    - test_type: 'main', 'subquestions', or 'full' (optional, defaults to 'main')
    
    Returns JSON response with task_id for polling.
    """
    try:
        # Get the question
        question = get_object_or_404(Question, id=question_id)
        
        # Parse request data
        try:
            data = json.loads(request.body) if request.body else {}
        except json.JSONDecodeError:
            return JsonResponse({
                'success': False,
                'error': 'Invalid JSON data',
                'error_type': 'validation'
            }, status=400)
        
        test_type = data.get('test_type', 'main')
        
        # Validate test_type
        valid_test_types = ['main', 'subquestions', 'full']
        if test_type not in valid_test_types:
            return JsonResponse({
                'success': False,
                'error': f'Invalid test_type. Must be one of: {valid_test_types}',
                'error_type': 'validation'
            }, status=400)
        
        # Initialize service with better error handling
        try:
            service = OpenAITestService()
            logger.info(f"OpenAITestService initialized. Debug: {getattr(service, 'debug_info', {})}")
        except Exception as e:
            from .services import OpenAIServiceError
            error_msg = str(e)
            debug_info = {}

            # Extract debug info if available
            if 'Debug:' in error_msg:
                try:
                    debug_part = error_msg.split('Debug:')[1].strip()
                    import ast
                    debug_info = ast.literal_eval(debug_part)
                except:
                    debug_info = {'raw_debug': debug_part}

            logger.error(f"Failed to initialize OpenAITestService: {e}")
            return JsonResponse({
                'success': False,
                'error': error_msg.split('Debug:')[0].strip() if 'Debug:' in error_msg else error_msg,
                'error_type': 'service_initialization',
                'debug_info': debug_info,
                'help': 'Check browser console (F12) for detailed debug information'
            }, status=503)

        # Check rate limits before creating task
        try:
            # Just check limits, don't actually run the test
            from .models import UserDailyLimit, QuestionDailyLimit

            # Check user daily limit
            user_limit = UserDailyLimit.get_or_create_today(request.user)
            if user_limit.remaining_tests <= 0:
                return JsonResponse({
                    'success': False,
                    'error': f'Daily limit reached. You have {user_limit.remaining_tests} tests remaining today.',
                    'error_type': 'rate_limit',
                    'remaining_limits': {
                        'user_daily': 0,
                        'question_daily': QuestionDailyLimit.get_or_create_today(request.user, question).remaining_tests,
                    }
                }, status=429)

            # Check question daily limit
            question_limit = QuestionDailyLimit.get_or_create_today(request.user, question)
            if question_limit.remaining_tests <= 0:
                return JsonResponse({
                    'success': False,
                    'error': f'Daily limit for this question reached. You have {question_limit.remaining_tests} tests remaining for this question today.',
                    'error_type': 'rate_limit',
                    'remaining_limits': {
                        'user_daily': user_limit.remaining_tests,
                        'question_daily': 0,
                    }
                }, status=429)
        except Exception as e:
            logger.warning(f"Error checking rate limits: {e}")
            # Continue anyway - let the async task handle it
        
        # Create TestAttempt record
        test_attempt = TestAttempt.objects.create(
            question=question,
            user=request.user,
            model_name='gpt-5',
            test_type=test_type,
            status='pending',
            prompt='',  # Will be filled by async task
            response='',  # Will be filled by async task
        )
        
        # Queue async task
        task_id = async_task(
            'ai_testing.services.async_test_question',
            test_attempt.id,
            task_name=f'test_question_{question_id}_{request.user.id}'
        )
        
        # Store task_id in TestAttempt
        test_attempt.task_id = task_id
        test_attempt.save(update_fields=['task_id'])
        
        logger.info(f"Queued async test task {task_id} for question {question_id} by user {request.user.id}")
        
        return JsonResponse({
            'success': True,
            'task_id': task_id,
            'test_attempt_id': test_attempt.id,
            'status': 'queued',
            'message': 'Test queued for processing. Poll the status endpoint for results.'
        }, status=202)  # 202 Accepted
        
    except Question.DoesNotExist:
        return JsonResponse({
            'success': False,
            'error': 'Question not found',
            'error_type': 'not_found'
        }, status=404)
        
    except Exception as e:
        logger.error(f"Unexpected error in test_question_ajax: {e}")
        return JsonResponse({
            'success': False,
            'error': 'An unexpected error occurred',
            'error_type': 'server_error'
        }, status=500)


@login_required
def get_user_limits(request):
    """
    Get user's current rate limit status.
    
    Returns JSON with remaining daily limits and reset times.
    """
    try:
        service = OpenAITestService()
        limits = service.get_user_remaining_limits(request.user)
        
        return JsonResponse({
            'success': True,
            'limits': limits
        })
        
    except Exception as e:
        logger.error(f"Error getting user limits: {e}")
        return JsonResponse({
            'success': False,
            'error': 'Failed to get rate limit information',
            'error_type': 'server_error'
        }, status=500)


@require_POST
@login_required
def cancel_test(request, test_attempt_id):
    """
    Cancel a running AI test.
    
    Stops the Django-Q task and updates the TestAttempt status.
    """
    try:
        # Get the test attempt
        test_attempt = get_object_or_404(TestAttempt, id=test_attempt_id, user=request.user)
        
        # Only allow cancelling if test is still processing
        if test_attempt.status not in ['pending', 'processing']:
            return JsonResponse({
                'success': False,
                'error': f'Cannot cancel test with status: {test_attempt.status}',
                'error_type': 'invalid_status'
            }, status=400)
        
        # Update the test attempt status
        test_attempt.status = 'cancelled'
        test_attempt.error_message = 'Test cancelled by user'
        test_attempt.save()
        
        # Try to delete the Django-Q task if it exists
        if test_attempt.task_id:
            try:
                from django_q.models import Task, OrmQ
                
                # Remove from queue if still pending
                OrmQ.objects.filter(key=test_attempt.task_id).delete()
                
                # Mark task as failed if it exists
                try:
                    task = Task.objects.get(id=test_attempt.task_id)
                    if not task.stopped:
                        # Task is still running, we can't directly stop it but marking TestAttempt as cancelled
                        # will prevent results from being saved
                        logger.info(f"Marked task {test_attempt.task_id} for cancellation")
                except Task.DoesNotExist:
                    pass
                    
            except Exception as e:
                logger.warning(f"Error cancelling Django-Q task: {e}")
        
        logger.info(f"Cancelled test attempt {test_attempt_id}")
        
        return JsonResponse({
            'success': True,
            'message': 'Test cancelled successfully'
        })
        
    except TestAttempt.DoesNotExist:
        return JsonResponse({
            'success': False,
            'error': 'Test attempt not found or access denied',
            'error_type': 'not_found'
        }, status=404)
        
    except Exception as e:
        logger.error(f"Error cancelling test: {e}")
        return JsonResponse({
            'success': False,
            'error': 'Failed to cancel test',
            'error_type': 'server_error'
        }, status=500)


@login_required
def diagnose_api(request):
    """
    Diagnostic endpoint to test OpenAI API connectivity and configuration.

    Returns detailed debug information about the service initialization.
    """
    import os
    import sys
    from pathlib import Path

    diagnostic_info = {
        'environment': {
            'python_version': sys.version,
            'django_settings': os.environ.get('DJANGO_SETTINGS_MODULE', 'NOT SET'),
            'working_directory': os.getcwd(),
            'user': request.user.username if request.user.is_authenticated else 'anonymous',
        },
        'service_initialization': {},
        'api_connectivity': False,
    }

    try:
        # Try to initialize the service
        service = OpenAITestService()
        diagnostic_info['service_initialization'] = getattr(service, 'debug_info', {})
        diagnostic_info['api_connectivity'] = bool(service.client)

        # Additional checks
        diagnostic_info['checks'] = {
            'has_api_key': bool(service.api_key),
            'has_client': bool(service.client),
            'api_key_length': len(service.api_key) if service.api_key else 0,
        }

        # Try a simple API call if client exists
        if service.client and service.api_key:
            try:
                # Try to list models as a connectivity test
                models = service.client.models.list()
                diagnostic_info['api_test'] = {
                    'status': 'success',
                    'models_available': True
                }
            except Exception as e:
                diagnostic_info['api_test'] = {
                    'status': 'failed',
                    'error': str(e)
                }

    except Exception as e:
        diagnostic_info['service_initialization']['error'] = str(e)
        diagnostic_info['service_initialization']['error_type'] = type(e).__name__

    return JsonResponse(diagnostic_info)


@login_required
def get_question_test_history(request, question_id):
    """
    Get recent test history for a question.
    
    Returns JSON with recent successful test attempts.
    """
    try:
        question = get_object_or_404(Question, id=question_id)
        service = OpenAITestService()
        
        # Get recent test history
        history = service.get_question_test_history(question, limit=5)
        
        # Format history for JSON response
        history_data = []
        for attempt in history:
            history_data.append({
                'id': attempt.id,
                'created_at': attempt.created_at.isoformat(),
                'model_name': attempt.model_name,
                'test_type': attempt.test_type,
                'response_time_ms': attempt.response_time_ms,
                'token_usage': {
                    'total_tokens': attempt.total_tokens,
                    'prompt_tokens': attempt.prompt_tokens,
                    'completion_tokens': attempt.completion_tokens,
                },
                'cost_estimate': attempt.cost_estimate,
                'response': attempt.response[:200] + '...' if len(attempt.response) > 200 else attempt.response,
            })
        
        return JsonResponse({
            'success': True,
            'history': history_data,
            'question_title': question.title,
        })
        
    except Question.DoesNotExist:
        return JsonResponse({
            'success': False,
            'error': 'Question not found',
            'error_type': 'not_found'
        }, status=404)
        
    except Exception as e:
        logger.error(f"Error getting question test history: {e}")
        return JsonResponse({
            'success': False,
            'error': 'Failed to get test history',
            'error_type': 'server_error'
        }, status=500)


@require_GET
@login_required
def check_test_status(request, test_attempt_id):
    """
    Check the status of an async test task.
    
    Returns JSON with current status and results if available.
    """
    try:
        # Get the test attempt
        test_attempt = get_object_or_404(TestAttempt, id=test_attempt_id, user=request.user)
        
        # Check if Django-Q task failed (timeout or error) but TestAttempt wasn't updated
        if test_attempt.status == 'processing' and test_attempt.task_id:
            try:
                from django_q.models import Task
                task = Task.objects.get(id=test_attempt.task_id)
                
                # If task failed (timeout or error), update the TestAttempt
                if not task.success and task.stopped:
                    test_attempt.status = 'timeout'
                    test_attempt.error_message = 'The AI test timed out while attempting to solve this problem. This can occur for particularly complex questions. Note that models in the actual benchmark evaluation will have significantly more time to work through challenging problems.'
                    test_attempt.save()
                    logger.info(f"Updated TestAttempt {test_attempt_id} to timeout based on failed Django-Q task")
            except Task.DoesNotExist:
                pass  # Task might not exist yet if it's still queued
        
        # Build response based on status
        response_data = {
            'success': True,
            'test_attempt_id': test_attempt.id,
            'status': test_attempt.status,
            'created_at': test_attempt.created_at.isoformat(),
        }
        
        # Add task info if available
        if test_attempt.task_id:
            response_data['task_id'] = test_attempt.task_id
            
            # Try to get task result from Django-Q
            try:
                from django_q.models import Task
                task = Task.objects.get(id=test_attempt.task_id)
                response_data['task_status'] = 'completed' if task.success else 'failed'
            except:
                # Task not found or not completed yet
                if test_attempt.status in ['pending', 'processing']:
                    response_data['task_status'] = 'processing'
                else:
                    response_data['task_status'] = 'unknown'
        
        # Add results if test is complete
        if test_attempt.status == 'success':
            response_data.update({
                'response': test_attempt.response,
                'response_time_ms': test_attempt.response_time_ms,
                'cost_estimate': test_attempt.cost_estimate,
                'token_usage': {
                    'total_tokens': test_attempt.total_tokens,
                    'prompt_tokens': test_attempt.prompt_tokens,
                    'completion_tokens': test_attempt.completion_tokens,
                },
                'model_name': test_attempt.model_name,
            })
            
            # Calculate tool usage if needed (for compatibility)
            response_data['tool_usage'] = {
                'web_search_calls': 0,
                'code_interpreter_sessions': 0,
            }
            
            # Get remaining limits
            service = OpenAITestService()
            limits = service.get_user_remaining_limits(request.user)
            response_data['remaining_limits'] = {
                'user_daily': limits['user_daily_remaining'],
                'question_daily': 5,  # TODO: get actual question limit
            }
            
        elif test_attempt.status in ['error', 'timeout', 'rate_limited', 'cancelled']:
            response_data['error'] = test_attempt.error_message or 'An error occurred'
            response_data['error_type'] = test_attempt.status
            
        return JsonResponse(response_data)
        
    except TestAttempt.DoesNotExist:
        return JsonResponse({
            'success': False,
            'error': 'Test attempt not found or access denied',
            'error_type': 'not_found'
        }, status=404)
        
    except Exception as e:
        logger.error(f"Error checking test status: {e}")
        return JsonResponse({
            'success': False,
            'error': 'Failed to check test status',
            'error_type': 'server_error'
        }, status=500)
