"""
Models for the review invitation system.

Provides data models for:
- Review invitations (internal and external reviewers)
- Review assignments tracking
- User notification preferences

Integrates with django-invitations for external reviewer workflow.
"""

from django.db import models
from django.conf import settings
from django.utils import timezone
from django.contrib.auth import get_user_model
from invitations.models import Invitation
import secrets
import string

User = get_user_model()


class ReviewInvitation(models.Model):
    """
    Extends django-invitations to track review-specific invitation data.
    We'll use django-invitations for the core token/email functionality,
    and this model for our review-specific needs.
    """
    
    STATUS_CHOICES = [
        ('pending', 'Pending'),
        ('accepted', 'Accepted'),
        ('declined', 'Declined'),
        ('expired', 'Expired'),
        ('revoked', 'Revoked'),
    ]
    
    # Link to django-invitations Invitation model (for external reviewers only)
    invitation = models.OneToOneField(
        Invitation,
        on_delete=models.CASCADE,
        null=True,
        blank=True,
        related_name='review_invitation'
    )
    
    # Custom token for internal reviewer invitations
    token = models.CharField(
        max_length=64, 
        unique=True, 
        null=True, 
        blank=True,
        help_text='Token for internal reviewer accept/decline links'
    )
    
    # Review-specific fields
    question = models.ForeignKey(
        'questions.Question',
        on_delete=models.CASCADE,
        related_name='review_invitations'
    )
    
    invited_by = models.ForeignKey(
        settings.AUTH_USER_MODEL,
        on_delete=models.SET_NULL,
        null=True,
        related_name='sent_review_invitations'
    )
    
    # For external reviewers who don't have accounts yet
    external_first_name = models.CharField(max_length=100, blank=True)
    external_last_name = models.CharField(max_length=100, blank=True)
    
    # Status tracking
    status = models.CharField(
        max_length=20,
        choices=STATUS_CHOICES,
        default='pending'
    )
    
    # Custom message for the invitation email
    custom_message = models.TextField(blank=True)
    
    # Track response
    first_responded_at = models.DateTimeField(null=True, blank=True)
    last_accessed_at = models.DateTimeField(null=True, blank=True)
    
    # If external user created an account
    created_account = models.ForeignKey(
        settings.AUTH_USER_MODEL,
        on_delete=models.SET_NULL,
        null=True,
        blank=True,
        related_name='created_from_invitation'
    )
    
    # Reminder tracking
    reminder_sent_at = models.DateTimeField(null=True, blank=True)
    
    # Timestamps
    created_at = models.DateTimeField(auto_now_add=True)
    updated_at = models.DateTimeField(auto_now=True)
    
    class Meta:
        db_table = 'review_invitations'
        ordering = ['-created_at']
        
    def __str__(self):
        if self.invitation and self.invitation.invitee:
            reviewer = self.invitation.invitee.get_full_name() or self.invitation.invitee.email
        elif self.external_first_name:
            reviewer = f"{self.external_first_name} {self.external_last_name}"
        else:
            reviewer = self.invitation.email if self.invitation else "Unknown"
        return f"Review invitation for {self.question.title[:50]} to {reviewer}"
    
    @property
    def is_expired(self):
        """Check if the invitation has expired (14 days)."""
        if self.status in ['accepted', 'declined', 'revoked']:
            return False
        expiry_date = self.created_at + timezone.timedelta(days=14)
        return timezone.now() > expiry_date
    
    @property
    def needs_reminder(self):
        """Check if a reminder should be sent (10 days old, no reminder sent yet)."""
        if self.status != 'pending' or self.reminder_sent_at:
            return False
        reminder_date = self.created_at + timezone.timedelta(days=10)
        return timezone.now() > reminder_date
    
    def mark_accessed(self):
        """Update access timestamp when invitation link is clicked."""
        self.last_accessed_at = timezone.now()
        if not self.first_responded_at:
            self.first_responded_at = timezone.now()
        self.save(update_fields=['last_accessed_at', 'first_responded_at'])
    
    @classmethod
    def generate_token(cls):
        """Generate a unique token for internal reviewer invitations."""
        alphabet = string.ascii_letters + string.digits
        while True:
            token = ''.join(secrets.choice(alphabet) for _ in range(32))
            if not cls.objects.filter(token=token).exists():
                return token


class ReviewAssignment(models.Model):
    """
    Tracks which users have been assigned to review which questions.
    This persists even after invitation expires/is declined.
    """
    user = models.ForeignKey(
        settings.AUTH_USER_MODEL,
        on_delete=models.CASCADE,
        related_name='review_assignments'
    )
    
    question = models.ForeignKey(
        'questions.Question',
        on_delete=models.CASCADE,
        related_name='reviewer_assignments'
    )
    
    invitation = models.ForeignKey(
        ReviewInvitation,
        on_delete=models.SET_NULL,
        null=True,
        blank=True,
        related_name='assignments'
    )
    
    # Track if this is an active assignment
    is_active = models.BooleanField(default=True)
    
    assigned_at = models.DateTimeField(auto_now_add=True)
    
    class Meta:
        db_table = 'review_assignments'
        unique_together = ['user', 'question']
        ordering = ['-assigned_at']
        
    def __str__(self):
        return f"{self.user.get_full_name() or self.user.email} assigned to review {self.question.title[:50]}"


class UserNotificationPreferences(models.Model):
    """
    User preferences for email notifications.
    """
    user = models.OneToOneField(
        settings.AUTH_USER_MODEL,
        on_delete=models.CASCADE,
        related_name='notification_preferences'
    )
    
    # Notification preferences
    notify_own_question_review = models.BooleanField(
        default=True,
        help_text="Receive notifications when your questions get reviewed"
    )
    notify_own_question_accepted = models.BooleanField(
        default=True,
        help_text="Receive notifications when your questions are accepted"
    )
    notify_review_reply = models.BooleanField(
        default=True,
        help_text="Receive notifications when authors reply to your reviews"
    )
    notify_own_question_grading = models.BooleanField(
        default=True,
        help_text="Receive notifications when AI answers to your questions are available for grading"
    )
    notify_reviewed_question_grading = models.BooleanField(
        default=False,  # Opt-in for external reviewers
        help_text="Receive notifications when AI answers to questions you reviewed are available for grading"
    )
    
    # Privacy preferences
    show_in_leaderboard = models.BooleanField(
        default=True,  # Will be set to False for external reviewers
        help_text="Display your name in public leaderboards"
    )
    
    created_at = models.DateTimeField(auto_now_add=True)
    updated_at = models.DateTimeField(auto_now=True)
    
    class Meta:
        db_table = 'user_notification_preferences'
        verbose_name_plural = "User notification preferences"
        
    def __str__(self):
        return f"Notification preferences for {self.user.get_full_name() or self.user.email}"
