from django.db import models
from participants.models import Participant


class QuestionState(models.Model):
    """
    Defines possible states in question lifecycle.
    
    Available states:
    - draft: Question is being written/edited
    - under_review: Question is being reviewed by peers
    - approved: Question has been approved for use
    - retracted: Question has been withdrawn from use
    - active: Question is marked for leaderboard inclusion (admin-only)
    """
    id = models.AutoField(primary_key=True)
    status = models.TextField(blank=True, null=True)
    
    class Meta:
        managed = False
        db_table = 'question_states'
    
    def __str__(self):
        return self.status or f"State {self.id}"


class Question(models.Model):
    """
    Main repository of benchmark problems.
    
    Contains LaTeX-formatted mathematical content and metadata about difficulty.
    The database has triggers for automatic timestamp updates.
    """
    id = models.AutoField(primary_key=True)
    title = models.TextField(blank=True, null=True)
    author = models.ForeignKey(Participant, on_delete=models.CASCADE, db_column='author')
    creation_time = models.DateTimeField(auto_now_add=True)  # Auto-set by Django on creation
    last_modified = models.DateTimeField(auto_now=True)     # Auto-updated by Django on save  
    difficulty_background = models.IntegerField(blank=True, null=True)  # 1-5 scale
    difficulty_reasoning = models.IntegerField(blank=True, null=True)   # 1-5 scale
    difficulty_insight = models.IntegerField(blank=True, null=True)     # 1-5 scale
    difficulty_compute = models.IntegerField(blank=True, null=True)     # 1-5 scale
    tags = models.TextField(blank=True, null=True)  # Comma-separated keywords
    text = models.TextField(blank=True, null=True)  # Question text (LaTeX formatted)
    solution = models.TextField(blank=True, null=True)  # Solution (LaTeX formatted)
    status = models.ForeignKey(QuestionState, on_delete=models.CASCADE, db_column='status')
    benchmark_inclusion = models.BooleanField(default=True)  # Include in benchmark evaluation results
    published = models.DateField(blank=True, null=True)  # Date when question was made public (NULL = unpublished, irreversible once set)

    class Meta:
        managed = False
        db_table = 'questions'
    
    def __str__(self):
        return self.title or f"Question {self.id}"
    
    @property
    def difficulty_levels(self):
        """Return a dict of all difficulty ratings."""
        return {
            'background': self.difficulty_background,
            'reasoning': self.difficulty_reasoning,
            'insight': self.difficulty_insight,
            'compute': self.difficulty_compute,
        }
    
    def get_open_reviews(self):
        """
        Get reviews that are open for discussion using new logic.
        Open = No replies OR (has replies + not concluded + decision not in ['accept', 'reject'])
        """
        from django.db.models import Q
        
        # Reviews without replies are always open
        no_replies = Q(replies__isnull=True)
        
        # Reviews with replies that aren't concluded and don't have final decisions
        actionable_with_replies = Q(
            replies__isnull=False,
            is_concluded=False
        ) & ~Q(decision__in=['accept', 'reject'])
        
        return self.reviews.filter(no_replies | actionable_with_replies).distinct()
    
    def get_closed_reviews(self):
        """
        Get reviews that are closed using new logic.
        Closed = Has replies AND (concluded OR decision in ['accept', 'reject'])
        """
        from django.db.models import Q
        
        # Has replies AND (concluded OR final decision made)
        return self.reviews.filter(
            replies__isnull=False
        ).filter(
            Q(is_concluded=True) | Q(decision__in=['accept', 'reject'])
        ).distinct()
    
    @property
    def open_review_count(self):
        """Count of open reviews using new logic."""
        return self.get_open_reviews().count()
    
    @property  
    def closed_review_count(self):
        """Count of closed reviews using new logic."""
        return self.get_closed_reviews().count()
    
    def get_absolute_url(self):
        """Return the URL for this question's detail page."""
        from django.urls import reverse
        return reverse('questions:detail', kwargs={'pk': self.pk})


class EvaluationType(models.Model):
    """
    Methods for evaluating subquestion answers (exact_match, numeric_tolerance, etc.).
    """
    id = models.AutoField(primary_key=True)
    evaluation_type = models.TextField(blank=True, null=True)
    
    class Meta:
        managed = False
        db_table = 'evaluation_types'
    
    def __str__(self):
        return self.evaluation_type or f"EvaluationType {self.id}"


class Subquestion(models.Model):
    """
    Automatically-gradable questions with unique answers.
    Uses string-based ordering system for flexible insertion.
    """
    id = models.AutoField(primary_key=True)
    question = models.ForeignKey(Question, on_delete=models.CASCADE, db_column='question_id')
    subquestion_order = models.TextField(blank=True, null=True)  # String-based ordering
    text = models.TextField(blank=True, null=True)
    answer = models.TextField(blank=True, null=True)  # Expected answer
    rationale = models.TextField(blank=True, null=True)  # Explanation
    points = models.IntegerField(blank=True, null=True)  # Must be positive
    evaluation_type = models.ForeignKey(EvaluationType, on_delete=models.CASCADE, db_column='evaluation_type_id')
    
    class Meta:
        managed = False
        db_table = 'subquestions'
    
    def __str__(self):
        return f"Subquestion {self.subquestion_order} of {self.question}"



class QuestionReview(models.Model):
    """
    Reviews for questions submitted to the benchmark.
    
    Tracks reviewer feedback, decisions, and anonymity preferences.
    Database triggers handle automatic timestamp updates.
    """
    # Review decision choices
    DECISION_CHOICES = [
        ('accept', 'Recommend for acceptance'),
        ('revise', 'Needs revision'),
        ('reject', 'Not suitable'),
    ]
    
    id = models.AutoField(primary_key=True)
    question = models.ForeignKey(Question, on_delete=models.CASCADE, db_column='question_id', related_name='reviews')
    reviewer = models.ForeignKey(Participant, on_delete=models.CASCADE, db_column='reviewer')
    time = models.DateTimeField(auto_now_add=True)  # Set by database trigger
    comment = models.TextField(blank=True, null=True)  # Markdown + LaTeX formatted feedback
    decision = models.CharField(max_length=10, choices=DECISION_CHOICES, blank=True, null=True)
    is_anonymous = models.BooleanField(default=False)  # Reviewer can choose anonymity
    is_concluded = models.BooleanField(default=False)  # Whether this review conversation is concluded
    
    class Meta:
        managed = False  # Don't let Django manage the table
        db_table = 'question_reviews'
        ordering = ['-time']  # Most recent reviews first
    
    def __str__(self):
        reviewer_name = "Anonymous" if self.is_anonymous else self.reviewer.name
        return f"Review by {reviewer_name} for {self.question.title}"
    
    @property
    def can_view(self, user):
        """Check if a user can view this review."""
        if not user.is_authenticated:
            return False
        
        # Author can always see reviews of their questions
        if hasattr(user, 'participant') and self.question.author == user.participant:
            return True
        
        # Reviewer can see their own reviews
        if hasattr(user, 'participant') and self.reviewer == user.participant:
            return True
        
        # Admins can see all reviews
        if user.is_superuser or (hasattr(user, 'participant') and user.participant.is_admin):
            return True
        
        return False
    
    def get_reviewer_display(self):
        """Get the display name for the reviewer, respecting anonymity."""
        if self.is_anonymous:
            return "Anonymous Reviewer"
        return self.reviewer.name


class ReviewReply(models.Model):
    """
    Represents an author's reply to a review.
    
    Authors can reply to reviews of their questions to clarify points
    or explain changes they've made in response to feedback.
    """
    
    id = models.AutoField(primary_key=True)
    review = models.ForeignKey(QuestionReview, on_delete=models.CASCADE, related_name='replies')
    author = models.ForeignKey(Participant, on_delete=models.CASCADE, db_column='author_id')
    comment = models.TextField()
    creation_time = models.DateTimeField(auto_now_add=True)
    
    class Meta:
        managed = False
        db_table = 'review_replies'
        ordering = ['creation_time']
        verbose_name_plural = "Review replies"

    def __str__(self):
        return f"Reply by {self.author.name} to review #{self.review.id}"


class QuestionVersion(models.Model):
    """
    Tracks version history for questions and subquestions.

    Every save to a question creates a new version record, capturing the complete
    state of the question and its subquestions at that point in time. This enables:
    - Viewing historical versions (similar to StackExchange edit history)
    - Detecting modifications after model evaluations
    - Diff comparison between versions

    Subquestions are stored as a JSON blob for atomic versioning - when you restore
    a version, you get the exact subquestions as they existed at that time.
    """
    import json

    id = models.AutoField(primary_key=True)
    question = models.ForeignKey(Question, on_delete=models.CASCADE, db_column='question_id', related_name='versions')
    version_number = models.IntegerField()
    created_at = models.DateTimeField(auto_now_add=True)
    created_by = models.ForeignKey(Participant, on_delete=models.SET_NULL, null=True, blank=True, db_column='created_by_id')

    # Snapshot of question fields at this version
    title = models.TextField()
    text = models.TextField()
    solution = models.TextField(blank=True, null=True)
    difficulty_background = models.IntegerField(blank=True, null=True)
    difficulty_reasoning = models.IntegerField(blank=True, null=True)
    difficulty_insight = models.IntegerField(blank=True, null=True)
    difficulty_compute = models.IntegerField(blank=True, null=True)
    tags = models.TextField(blank=True, null=True)

    # JSON blob storing subquestions at this version
    # Format: [{"order": "a", "text": "...", "answer": "...", "rationale": "...", "points": 1, "evaluation_type": "..."}]
    subquestions_json = models.TextField(blank=True, null=True)

    class Meta:
        managed = False  # Table created by migration with raw SQL
        db_table = 'question_versions'
        ordering = ['-version_number']
        unique_together = [['question', 'version_number']]

    def __str__(self):
        return f"Version {self.version_number} of Question {self.question_id}"

    @classmethod
    def create_version(cls, question, created_by=None):
        """
        Create a new version record capturing the current state of a question.

        Args:
            question: The Question instance to snapshot
            created_by: The Participant who made this change (optional)

        Returns:
            The newly created QuestionVersion instance
        """
        import json

        # Get the next version number
        latest_version = cls.objects.filter(question=question).order_by('-version_number').first()
        next_version = (latest_version.version_number + 1) if latest_version else 1

        # Serialize subquestions to JSON
        subquestions_data = []
        for sq in question.subquestion_set.all().order_by('subquestion_order'):
            subquestions_data.append({
                'order': sq.subquestion_order,
                'text': sq.text,
                'answer': sq.answer,
                'rationale': sq.rationale,
                'points': sq.points,
                'evaluation_type': sq.evaluation_type.evaluation_type if sq.evaluation_type else None,
            })

        # Create the version record
        version = cls.objects.create(
            question=question,
            version_number=next_version,
            created_by=created_by,
            title=question.title or '',
            text=question.text or '',
            solution=question.solution,
            difficulty_background=question.difficulty_background,
            difficulty_reasoning=question.difficulty_reasoning,
            difficulty_insight=question.difficulty_insight,
            difficulty_compute=question.difficulty_compute,
            tags=question.tags,
            subquestions_json=json.dumps(subquestions_data) if subquestions_data else None,
        )

        return version

    def get_subquestions(self):
        """
        Parse and return the subquestions from the JSON blob.

        Returns:
            List of dicts with subquestion data, or empty list if none
        """
        import json

        if not self.subquestions_json:
            return []
        try:
            return json.loads(self.subquestions_json)
        except json.JSONDecodeError:
            return []

    @classmethod
    def get_latest_evaluation_time(cls, question):
        """
        Get the timestamp of the most recent model evaluation for a question.

        Returns:
            datetime or None if no evaluations exist
        """
        from model_evaluation.models import ModelAttempt

        latest = ModelAttempt.objects.filter(
            question_id=question.id
        ).order_by('-time').first()

        return latest.time if latest else None

    @classmethod
    def was_modified_after_evaluation(cls, question):
        """
        Check if a question was modified after its most recent evaluation.

        Returns:
            bool: True if modified after last evaluation, False otherwise
        """
        latest_eval_time = cls.get_latest_evaluation_time(question)

        if not latest_eval_time:
            # No evaluations yet, so no conflict possible
            return False

        # Compare question's last_modified with latest evaluation time
        return question.last_modified > latest_eval_time
