"""
OpenAI API integration service for question testing.

Provides rate-limited AI testing functionality with proper error handling
and database tracking.
"""

import time
import logging
from datetime import datetime
from typing import Dict, Optional, Tuple

from django.conf import settings
from django.db import transaction
from decouple import config

from .models import TestAttempt, UserDailyLimit, QuestionDailyLimit
from questions.models import Question


# Configure logging with more verbose output
logger = logging.getLogger(__name__)
logger.setLevel(logging.INFO)

# Add console handler if not already present
if not logger.handlers:
    console_handler = logging.StreamHandler()
    console_handler.setLevel(logging.INFO)
    formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')
    console_handler.setFormatter(formatter)
    logger.addHandler(console_handler)


class OpenAIServiceError(Exception):
    """Base exception for OpenAI service errors."""
    pass


class OpenAITestService:
    """
    Service class for testing questions with OpenAI API.
    
    Features:
    - Rate limiting (5 per question daily, 20 global daily per user)
    - Comprehensive error handling with retry logic
    - Database tracking of all attempts
    - Cost estimation based on token usage
    - Designed for easy extension to streaming
    """
    
    def __init__(self):
        """Initialize OpenAI test service with comprehensive error handling."""
        import os
        import sys
        try:
            # Try to get API key from decouple, fallback to environment variable
            self.api_key = config('OPENAI_API_KEY', default='')
            if not self.api_key:
                self.api_key = os.environ.get('OPENAI_API_KEY', '')
        except Exception as e:
            self.api_key = ''
            logger.error(f"Error getting API key: {e}")

        self.model_name = 'gpt-5'
        self.client = None

        if not self.api_key or self.api_key == 'your-api-key-here':
            logger.warning("OpenAI API key not configured")
        else:
            self._init_client()

    def _init_client(self):
        """Initialize OpenAI client."""
        try:
            from openai import OpenAI

            # Import error types for better error handling
            global APITimeoutError, RateLimitError, APIConnectionError, ServiceUnavailableError, InvalidRequestError
            try:
                from openai import APITimeoutError, RateLimitError, APIConnectionError, ServiceUnavailableError, InvalidRequestError
            except ImportError:
                # Fallback for older openai library versions
                APITimeoutError = RateLimitError = APIConnectionError = ServiceUnavailableError = InvalidRequestError = Exception

            # Create client with 30-minute timeout to match Django-Q
            self.client = OpenAI(api_key=self.api_key, timeout=1800.0)

        except ImportError as e:
            logger.error(f"OpenAI library import failed: {e}")
            raise OpenAIServiceError(f"OpenAI library not available: {e}")
        except Exception as e:
            logger.error(f"Failed to initialize OpenAI client: {type(e).__name__}: {e}")
            raise OpenAIServiceError(f"Client initialization failed: {e}")
    
    def check_rate_limits(self, user, question: Question) -> Tuple[bool, str]:
        """
        Check if user can make another test request.
        
        Args:
            user: Django user object
            question: Question to test
            
        Returns:
            Tuple of (can_proceed, error_message)
        """
        # Check global daily limit (20 for regular users, 100 for admins)
        user_limit = UserDailyLimit.get_or_create_today(user)
        if user_limit.remaining_tests <= 0:
            daily_limit = user_limit.get_daily_limit()
            return False, f"Daily limit exceeded. You have used all {daily_limit} tests today. Resets at midnight UTC."
        
        # Check per-question daily limit (5 per question per day)
        question_limit = QuestionDailyLimit.get_or_create_today(user, question)
        if question_limit.remaining_tests <= 0:
            return False, f"Question limit exceeded. You have tested this question 5 times today. Try again tomorrow."
        
        return True, ""
    
    def _build_question_prompt(self, question: Question) -> str:
        """
        Build the prompt to send to OpenAI for question solving.
        
        Args:
            question: Question object for AI to attempt solving
            
        Returns:
            Formatted prompt string
        """
        prompt_parts = []
        
        # Question content
        if question.title:
            prompt_parts.append(question.title)
            prompt_parts.append("")
        
        if question.text:
            prompt_parts.append(question.text)
            prompt_parts.append("")
        
        # Simple instruction
        prompt_parts.append("Please solve this mathematical problem step by step.")
        prompt_parts.append("")
        prompt_parts.append("FORMATTING INSTRUCTIONS:")
        prompt_parts.append("- Use standard HTML to be displayed as part of a website.")
        prompt_parts.append("- Use KaTeX for math: write valid KaTeX to render mathematics, \\( and \\) for inline, \\[ and \\] for display style.")
        prompt_parts.append("- Do NOT use custom LaTeX macros like \\BM, \\PP - spell them out as \\mathrm{BM}, \\mathbb{P}")
        prompt_parts.append("- Use standard LaTeX commands only (\\mathrm{}, \\mathbb{}, \\mathcal{}, etc.)")
        
        return "\n".join(prompt_parts)

    def _calculate_cost(self, prompt_tokens: int, completion_tokens: int, cached_tokens: int = 0) -> float:
        """
        Calculate cost based on GPT-5 token pricing.

        Args:
            prompt_tokens: Input tokens used
            completion_tokens: Output tokens used
            cached_tokens: Cached input tokens (if any)

        Returns:
            Estimated cost in USD
        """
        # GPT-5 pricing (as of August 2025)
        input_rate = 1.25e-6  # $1.25 per million tokens
        cached_rate = 0.13e-6  # $0.13 per million cached tokens
        output_rate = 10.00e-6  # $10.00 per million tokens

        # Calculate token costs
        regular_input_tokens = max(0, prompt_tokens - cached_tokens)
        total_cost = (
            regular_input_tokens * input_rate +
            cached_tokens * cached_rate +
            completion_tokens * output_rate
        )

        return total_cost
    
    def get_question_test_history(self, question: Question, limit: int = 5):
        """
        Get recent test history for a question.
        
        Args:
            question: Question object
            limit: Maximum number of results to return
            
        Returns:
            QuerySet of recent TestAttempt objects
        """
        return TestAttempt.objects.filter(
            question=question,
            status='success'
        ).order_by('-created_at')[:limit]
    
    def get_latest_successful_test(self, question: Question):
        """
        Get the most recent successful AI test for a question.
        
        Args:
            question: Question object
            
        Returns:
            TestAttempt object or None if no successful tests exist
        """
        try:
            return TestAttempt.objects.filter(
                question=question,
                status='success'
            ).order_by('-created_at').first()
        except TestAttempt.DoesNotExist:
            return None
    
    def get_user_remaining_limits(self, user):
        """
        Get user's remaining daily limits.
        
        Args:
            user: Django user object
            
        Returns:
            Dictionary with remaining limits information
        """
        user_limit = UserDailyLimit.get_or_create_today(user)
        
        daily_limit = user_limit.get_daily_limit()
        return {
            'user_daily_remaining': user_limit.remaining_tests,
            'user_daily_total': daily_limit,
            'reset_time': user_limit.reset_time.isoformat(),
        }


# Async task function for Django-Q
def async_test_question(test_attempt_id: int) -> Dict:
    """
    Background task for testing a question with OpenAI API.
    
    This function is executed by Django-Q workers asynchronously.
    It updates the TestAttempt record with results as processing occurs.
    
    Args:
        test_attempt_id: ID of the TestAttempt to process
        
    Returns:
        Dictionary with test results
    """
    # Import models here to avoid circular imports
    from .models import TestAttempt, UserDailyLimit, QuestionDailyLimit
    
    try:
        # Get the test attempt
        test_attempt = TestAttempt.objects.get(id=test_attempt_id)
        
        # Check if already cancelled
        if test_attempt.status == 'cancelled':
            logger.info(f"Test attempt {test_attempt_id} was cancelled before processing")
            return {
                'success': False,
                'test_attempt_id': test_attempt_id,
                'error': 'Test cancelled by user',
                'error_type': 'cancelled'
            }
        
        # Update status to processing
        test_attempt.status = 'processing'
        test_attempt.save(update_fields=['status'])
        
        # Get user and question
        user = test_attempt.user
        question = test_attempt.question

        # Create service instance
        service = OpenAITestService()
        
        # Validate API setup
        if not service.client:
            test_attempt.status = 'error'
            test_attempt.error_message = 'OpenAI API not configured. Please check your API key.'
            test_attempt.save()
            return {
                'success': False,
                'test_attempt_id': test_attempt.id,
                'error': 'OpenAI API not configured. Please check your API key.',
                'error_type': 'configuration'
            }
        
        # Check rate limits
        can_proceed, limit_error = service.check_rate_limits(user, question)
        if not can_proceed:
            test_attempt.status = 'rate_limited'
            test_attempt.error_message = limit_error
            test_attempt.save()
            
            return {
                'success': False,
                'test_attempt_id': test_attempt.id,
                'error': limit_error,
                'error_type': 'rate_limit'
            }
        
        try:
            # Build prompt
            prompt = service._build_question_prompt(question)
            test_attempt.prompt = prompt
            test_attempt.save(update_fields=['prompt'])
            
            # Make API call with timing
            start_time = time.time()
            
            # System instruction to guide tool usage
            system_instruction = (
                "You are an expert mathematician. Solve the given problem clearly and thoroughly. "
                "Only use web search when you need fresh facts or citations. Show the queries you ran. "
                "Use Code Interpreter for any non-trivial math, data wrangling, or plotting. "
                "Always return a concise reasoning summary."
            )
            
            # Check if cancelled before making expensive API call
            test_attempt.refresh_from_db()
            if test_attempt.status == 'cancelled':
                logger.info(f"Test attempt {test_attempt_id} cancelled before API call")
                return {
                    'success': False,
                    'test_attempt_id': test_attempt_id,
                    'error': 'Test cancelled by user',
                    'error_type': 'cancelled'
                }
            
            # Create response with tools
            response = service.client.responses.create(
                model=service.model_name,
                input=[
                    {"role": "developer", "content": [{"type": "input_text", "text": system_instruction}]},
                    {"role": "user", "content": [{"type": "input_text", "text": prompt}]}
                ],
                tools=[
                    {"type": "web_search"},  # Updated from web_search_preview for GPT-5
                    {"type": "code_interpreter", "container": {"type": "auto"}}
                ],
                reasoning={"effort": "high"},  # GPT-5 uses this parameter structure
                max_output_tokens=100000,  # Generous limit for complex problems
                background=False,  # Synchronous for now
                store=False  # Don't store artifacts
            )
            
            end_time = time.time()
            response_time_ms = int((end_time - start_time) * 1000)
            
            # Extract response data
            response_text = response.output_text if hasattr(response, 'output_text') else ""
            
            # Extract token usage from Responses API
            prompt_tokens = completion_tokens = total_tokens = 0
            cached_tokens = reasoning_tokens = 0
            
            if hasattr(response, 'usage') and response.usage:
                # Try multiple possible field names for token counts
                prompt_tokens = (
                    getattr(response.usage, 'prompt_tokens', 0) or 
                    getattr(response.usage, 'input_tokens', 0) or 
                    0
                )
                completion_tokens = (
                    getattr(response.usage, 'completion_tokens', 0) or 
                    getattr(response.usage, 'output_tokens', 0) or 
                    0
                )
                total_tokens = getattr(response.usage, 'total_tokens', 0) or (prompt_tokens + completion_tokens)
                
                # Get cached and reasoning tokens if available
                if hasattr(response.usage, 'input_tokens_details'):
                    cached_tokens = getattr(response.usage.input_tokens_details, 'cached_tokens', 0)
                if hasattr(response.usage, 'output_tokens_details'):
                    reasoning_tokens = getattr(response.usage.output_tokens_details, 'reasoning_tokens', 0)
            
            # Detect tool usage from response output
            tool_usage = {
                'web_search_calls': 0,
                'code_interpreter_sessions': 0,
            }
            
            # Count tool usage if available in response
            if hasattr(response, 'output') and response.output:
                for item in response.output:
                    if hasattr(item, 'type'):
                        if item.type == 'function_call' and hasattr(item, 'name'):
                            if 'web_search' in item.name.lower() or 'search' in item.name.lower():
                                tool_usage['web_search_calls'] += 1
                            elif 'code' in item.name.lower() or 'python' in item.name.lower():
                                tool_usage['code_interpreter_sessions'] = 1
            
            # Calculate cost
            token_usage = {
                'prompt_tokens': prompt_tokens,
                'completion_tokens': completion_tokens,
                'total_tokens': total_tokens,
                'cached_tokens': cached_tokens,
                'reasoning_tokens': reasoning_tokens
            }
            cost_estimate = service._calculate_cost(
                prompt_tokens,
                completion_tokens,
                cached_tokens
            )
            
            # Update test attempt with results
            test_attempt.response = response_text
            test_attempt.response_time_ms = response_time_ms
            test_attempt.prompt_tokens = prompt_tokens
            test_attempt.completion_tokens = completion_tokens
            test_attempt.total_tokens = total_tokens
            test_attempt.status = 'success'
            test_attempt.save()
            
            # Decrement rate limits
            user_limit = UserDailyLimit.get_or_create_today(user)
            user_limit.test_count += 1
            user_limit.save()
            
            question_limit = QuestionDailyLimit.get_or_create_today(user, question)
            question_limit.test_count += 1
            question_limit.save()
            
            logger.info(f"Async test completed successfully for question {question.id}")
            
            return {
                'success': True,
                'test_attempt_id': test_attempt.id,
                'response': response_text,
                'response_time_ms': response_time_ms,
                'cost_estimate': cost_estimate,
                'token_usage': token_usage,
                'tool_usage': tool_usage,
                'api_used': 'responses',
                'remaining_limits': {
                    'user_daily': user_limit.remaining_tests,
                    'question_daily': question_limit.remaining_tests,
                }
            }
            
        except Exception as e:
            # Handle API errors
            error_message = str(e)
            test_attempt.status = 'error'
            test_attempt.error_message = error_message
            test_attempt.save()
            
            logger.error(f"Error in async_test_question API call: {error_message}")
            
            # Check if it's a timeout error
            if 'timeout' in error_message.lower():
                test_attempt.status = 'timeout'
            
            test_attempt.save()
            
            return {
                'success': False,
                'test_attempt_id': test_attempt.id,
                'error': error_message,
                'error_type': 'api_error'
            }
            
    except TestAttempt.DoesNotExist:
        logger.error(f"TestAttempt {test_attempt_id} not found")
        return {
            'success': False,
            'error': f'TestAttempt {test_attempt_id} not found',
            'error_type': 'not_found'
        }
    except Exception as e:
        logger.error(f"Unexpected error in async_test_question: {str(e)}")
        
        # Update test attempt with error
        if 'test_attempt' in locals():
            test_attempt.status = 'error'
            test_attempt.error_message = str(e)
            test_attempt.save()
        
        return {
            'success': False,
            'error': str(e),
            'error_type': 'unexpected_error'
        }
