from django.db import models
from django.conf import settings
from django.utils import timezone
from datetime import date, datetime, timedelta
from zoneinfo import ZoneInfo
from questions.models import Question


class TestAttempt(models.Model):
    """
    Records each AI test attempt on a question for analysis and rate limiting.
    
    Tracks test attempts with their results and metadata for:
    - Analytics on AI performance
    - Rate limiting enforcement
    - Historical test results display
    """
    question = models.ForeignKey(Question, on_delete=models.CASCADE, related_name='test_attempts')
    user = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.CASCADE, related_name='test_attempts')
    created_at = models.DateTimeField(auto_now_add=True)
    
    # Test metadata
    model_name = models.CharField(max_length=100, default='gpt-5')
    test_type = models.CharField(
        max_length=20,
        choices=[
            ('main', 'Main Question'),
            ('subquestions', 'Subquestions'),
            ('full', 'Full Question + Subquestions'),
        ],
        default='main'
    )
    
    # Request and response data
    prompt = models.TextField(help_text="The prompt sent to the AI model")
    response = models.TextField(help_text="The AI model's response")
    response_time_ms = models.IntegerField(null=True, blank=True, help_text="Response time in milliseconds")
    
    # Token usage for cost tracking
    prompt_tokens = models.IntegerField(null=True, blank=True)
    completion_tokens = models.IntegerField(null=True, blank=True)
    total_tokens = models.IntegerField(null=True, blank=True)
    
    # Status tracking
    status = models.CharField(
        max_length=20,
        choices=[
            ('pending', 'Pending'),
            ('processing', 'Processing'),  # Added for async processing
            ('success', 'Success'),
            ('error', 'Error'),
            ('timeout', 'Timeout'),
            ('rate_limited', 'Rate Limited'),
            ('cancelled', 'Cancelled'),  # User cancelled the test
        ],
        default='pending'
    )
    error_message = models.TextField(null=True, blank=True)
    
    # Async task tracking
    task_id = models.CharField(max_length=128, null=True, blank=True, db_index=True,
                              help_text="Django-Q task ID for async processing")
    
    # Flag for test attempts (as mentioned in requirements)
    is_test_attempt = models.BooleanField(default=True)
    
    class Meta:
        ordering = ['-created_at']
        indexes = [
            models.Index(fields=['question', 'created_at']),
            models.Index(fields=['user', 'created_at']),
            models.Index(fields=['created_at']),
        ]
    
    def __str__(self):
        return f"Test of {self.question.title[:50]} by {self.user.email} at {self.created_at}"
    
    @property
    def cost_estimate(self):
        """Estimate cost based on token usage (GPT-5 pricing)."""
        if not self.total_tokens:
            return None
        
        # GPT-5 pricing: $1.25 per million input, $10.00 per million output tokens
        prompt_tokens = self.prompt_tokens or 0
        completion_tokens = self.completion_tokens or 0
        
        # If we have breakdown, use it for accurate cost calculation
        if prompt_tokens > 0 or completion_tokens > 0:
            input_cost = prompt_tokens * 1.25 / 1_000_000
            output_cost = completion_tokens * 10.00 / 1_000_000
            return input_cost + output_cost
        
        # Fallback: if no breakdown but we have total tokens, estimate cost
        # Use conservative assumption: 30% prompt, 70% completion
        estimated_prompt = int(self.total_tokens * 0.3)
        estimated_completion = int(self.total_tokens * 0.7)
        input_cost = estimated_prompt * 1.25 / 1_000_000
        output_cost = estimated_completion * 10.00 / 1_000_000
        return input_cost + output_cost


class UserDailyLimit(models.Model):
    """
    Tracks daily API usage limits per user.
    
    Enforces global daily limit of 20 tests per user.
    Resets at midnight UTC.
    """
    user = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.CASCADE, related_name='daily_limits')
    date = models.DateField(default=date.today)
    test_count = models.IntegerField(default=0)
    
    class Meta:
        unique_together = ['user', 'date']
        indexes = [
            models.Index(fields=['user', 'date']),
        ]
    
    def __str__(self):
        daily_limit = self.get_daily_limit()
        return f"{self.user.email} - {self.date}: {self.test_count}/{daily_limit} tests"
    
    @classmethod
    def get_or_create_today(cls, user):
        """Get or create today's limit record for a user."""
        today = date.today()
        limit, created = cls.objects.get_or_create(
            user=user,
            date=today,
            defaults={'test_count': 0}
        )
        return limit
    
    @property
    def remaining_tests(self):
        """Calculate remaining tests for today."""
        daily_limit = self.get_daily_limit()
        return max(0, daily_limit - self.test_count)
    
    def get_daily_limit(self):
        """Get the daily limit for this user based on their role."""
        try:
            if hasattr(self.user, 'participant') and self.user.participant.is_admin:
                return 100  # Admins get 100 tests per day
            else:
                return 20   # Regular users get 20 tests per day
        except:
            return 20       # Fallback to regular limit if participant not found
    
    @property
    def reset_time(self):
        """Calculate when the limit resets (midnight UTC)."""
        tomorrow = self.date + timedelta(days=1)
        midnight_utc = datetime.combine(tomorrow, datetime.min.time(), tzinfo=ZoneInfo('UTC'))
        return midnight_utc


class QuestionDailyLimit(models.Model):
    """
    Tracks daily API usage limits per question per user.
    
    Enforces limit of 5 tests per question per user per day.
    Resets at midnight UTC.
    """
    user = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.CASCADE)
    question = models.ForeignKey(Question, on_delete=models.CASCADE)
    date = models.DateField(default=date.today)
    test_count = models.IntegerField(default=0)
    
    class Meta:
        unique_together = ['user', 'question', 'date']
        indexes = [
            models.Index(fields=['user', 'question', 'date']),
        ]
    
    def __str__(self):
        return f"{self.user.email} - {self.question.title[:30]} - {self.date}: {self.test_count}/5 tests"
    
    @classmethod
    def get_or_create_today(cls, user, question):
        """Get or create today's limit record for a user and question."""
        today = date.today()
        limit, created = cls.objects.get_or_create(
            user=user,
            question=question,
            date=today,
            defaults={'test_count': 0}
        )
        return limit
    
    @property
    def remaining_tests(self):
        """Calculate remaining tests for this question today."""
        return max(0, 5 - self.test_count)
