"""
Views for participant management.

Provides CRUD operations for participant records with:
- Admin-only access for create/update/delete operations
- User account creation and linking
- Email verification workflow integration
- Model answer statistics for question authors
"""

from django.views.generic import ListView, CreateView, UpdateView, DeleteView
from django.db.models import Q, Count, Case, When, IntegerField
from django.urls import reverse_lazy
from django.contrib import messages
from django.db import IntegrityError, connection
from django.contrib.auth import get_user_model
from django.http import JsonResponse
from allauth.account.models import EmailAddress
from allauth.account.internal.flows.email_verification import send_verification_email_to_address
from .models import Participant
from .forms import ParticipantForm
from accounts.permissions import AdminRequiredMixin, AdminRequiredSignupRedirectMixin
from django.contrib.auth.mixins import LoginRequiredMixin

User = get_user_model()


class ParticipantListView(LoginRequiredMixin, ListView):
    """
    Display a list of all participants with search functionality.
    
    Features:
    - Bootstrap-styled table display
    - Search by name
    - Clean, professional appearance
    - Model error rate calculation
    """
    model = Participant
    template_name = 'participants/list.html'
    context_object_name = 'participants'
    paginate_by = 20  # Pagination for large datasets
    
    def get_queryset(self):
        """Filter participants based on search query and add question count annotations."""
        # Store stats in instance variable for later use
        self.model_answer_stats = self._get_model_answer_stats()
        
        # Handle sorting (default: total descending)
        sort_by = self.request.GET.get('sort', 'total')
        order = self.request.GET.get('order', 'desc')
        
        # Search functionality
        search_query = self.request.GET.get('search', '').strip()
        
        # If sorting by model_stumps (or legacy incorrect_percentage), use raw SQL for complete sorting
        if sort_by in ['model_stumps', 'incorrect_percentage']:
            # Use raw SQL to get properly sorted IDs
            with connection.cursor() as cursor:
                search_clause = ""
                search_params = []
                if search_query:
                    search_clause = "WHERE p.name LIKE ?"
                    search_params = [f'%{search_query}%']
                
                order_clause = "DESC" if order == 'desc' else "ASC"
                
                cursor.execute(f"""
                    SELECT p.id
                    FROM participants p
                    LEFT JOIN (
                        SELECT 
                            p2.id as participant_id,
                            COUNT(DISTINCT CASE WHEN msa.is_correct = 0 THEN msa.id END) as incorrect_count
                        FROM participants p2
                        LEFT JOIN questions q ON q.author = p2.id
                        LEFT JOIN question_states qs ON q.status = qs.id AND qs.status IN ('approved', 'active')
                        LEFT JOIN subquestions sq ON sq.question_id = q.id
                        LEFT JOIN model_subquestion_answers msa ON msa.subquestion_id = sq.id
                        LEFT JOIN models m ON msa.model_id = m.id
                        LEFT JOIN model_tiers mt ON m.tier_id = mt.id
                        WHERE mt.tier_number = 1 OR mt.tier_number IS NULL
                        GROUP BY p2.id
                    ) stats ON stats.participant_id = p.id
                    {search_clause}
                    ORDER BY 
                        stats.incorrect_count {order_clause} NULLS LAST
                """, search_params)
                
                sorted_ids = [row[0] for row in cursor.fetchall()]
            
            # Create queryset with the sorted IDs
            if sorted_ids:
                # Preserve the order from SQL
                preserved = Case(*[When(pk=pk, then=pos) for pos, pk in enumerate(sorted_ids)])
                queryset = Participant.objects.filter(id__in=sorted_ids).order_by(preserved)
            else:
                queryset = Participant.objects.none()
        else:
            # Regular Django ORM query for other sorting
            queryset = Participant.objects.all()
            
            # Search functionality
            if search_query:
                queryset = queryset.filter(Q(name__icontains=search_query))
        
        # Add annotations for question counts
        queryset = queryset.annotate(
            under_review_count=Count(
                Case(
                    When(question__status__status='under_review', then=1),
                    output_field=IntegerField()
                )
            ),
            approved_count=Count(
                Case(
                    When(Q(question__status__status='approved') | Q(question__status__status='active'), then=1),
                    output_field=IntegerField()
                )
            )
        ).annotate(
            total_count=Count(
                Case(
                    When(Q(question__status__status='under_review') | Q(question__status__status='approved') | Q(question__status__status='active'), then=1),
                    output_field=IntegerField()
                )
            )
        )
        
        # Handle sorting for non-model_stumps fields
        if sort_by not in ['model_stumps', 'incorrect_percentage']:
            sort_fields = {
                'name': 'name',
                'under_review': 'under_review_count',
                'approved': 'approved_count',
                'total': 'total_count'
            }
            
            if sort_by in sort_fields:
                order_by = sort_fields[sort_by]
                if order == 'desc':
                    order_by = f'-{order_by}'
                queryset = queryset.order_by(order_by)
            else:
                # Default fallback
                queryset = queryset.order_by('-total_count')
        
        return queryset
    
    def _get_model_answer_stats(self):
        """Get model answer statistics using raw SQL - Tier 1 models only."""
        with connection.cursor() as cursor:
            cursor.execute("""
                SELECT 
                    p.id,
                    COUNT(DISTINCT sq.id) as total_subquestions,
                    COUNT(DISTINCT msa.id) as total_model_answers,
                    COUNT(DISTINCT CASE WHEN msa.is_correct = 0 THEN msa.id END) as incorrect_answers,
                    CASE 
                        WHEN COUNT(DISTINCT msa.id) > 0 
                        THEN ROUND(100.0 * COUNT(DISTINCT CASE WHEN msa.is_correct = 0 THEN msa.id END) / COUNT(DISTINCT msa.id), 2)
                        ELSE NULL 
                    END as incorrect_percentage
                FROM participants p
                LEFT JOIN questions q ON q.author = p.id
                LEFT JOIN question_states qs ON q.status = qs.id AND qs.status IN ('approved', 'active')
                LEFT JOIN subquestions sq ON sq.question_id = q.id
                LEFT JOIN model_subquestion_answers msa ON msa.subquestion_id = sq.id
                LEFT JOIN models m ON msa.model_id = m.id
                LEFT JOIN model_tiers mt ON m.tier_id = mt.id
                WHERE mt.tier_number = 1 OR msa.id IS NULL
                GROUP BY p.id
            """)
            return {row[0]: {
                'total_subquestions': row[1],
                'total_model_answers': row[2],
                'incorrect_answers': row[3],
                'incorrect_percentage': row[4]
            } for row in cursor.fetchall()}
    
    def get_context_data(self, **kwargs):
        """Add search query, sorting parameters, and model answer stats to context."""
        context = super().get_context_data(**kwargs)
        context['search_query'] = self.request.GET.get('search', '')
        context['current_sort'] = self.request.GET.get('sort', 'total')
        context['current_order'] = self.request.GET.get('order', 'desc')
        
        # Add model answer stats to the paginated participants
        if 'participants' in context and hasattr(self, 'model_answer_stats'):
            participants = context['participants']
            for participant in participants:
                stats = self.model_answer_stats.get(participant.id, {})
                participant.total_subquestions = stats.get('total_subquestions', 0)
                participant.total_model_answers = stats.get('total_model_answers', 0)
                participant.incorrect_answers = stats.get('incorrect_answers', 0)
                participant.incorrect_percentage = stats.get('incorrect_percentage', None)
        
        return context


class ParticipantCreateView(AdminRequiredSignupRedirectMixin, CreateView):
    """
    View for adding new participants.
    
    Features:
    - Crispy forms with Bootstrap styling
    - Client-side and server-side validation
    - Success messages and error handling
    - Redirect back to participant list
    """
    model = Participant
    form_class = ParticipantForm
    template_name = 'participants/form.html'
    success_url = reverse_lazy('participants:list')
    
    def get_form_kwargs(self):
        """Pass the current user to the form for role-based field display."""
        kwargs = super().get_form_kwargs()
        kwargs['user'] = self.request.user
        return kwargs
    
    def form_valid(self, form):
        """Handle successful form submission with User account creation."""
        try:
            # Save the participant first
            response = super().form_valid(form)
            
            # Create User account if email is provided and no user is linked
            email = form.cleaned_data.get('email')
            if email and not self.object.user:
                self.create_user_account(email)
            
            # Success message
            action_msg = f'✅ Participant "{self.object.name}" has been added successfully!'
            if email:
                if self.object.user:
                    action_msg += f' User account created for {email}.'
                else:
                    action_msg += f' Email {email} added.'
            
            messages.success(self.request, action_msg)
            return response
            
        except IntegrityError:
            messages.error(
                self.request,
                '❌ Unable to save participant. Please check for duplicate names or emails.'
            )
            return self.form_invalid(form)
    
    def create_user_account(self, email):
        """Create a User account and link it to the participant."""
        try:
            # Check if a user with this email already exists
            existing_user = User.objects.filter(email__iexact=email).first()
            
            if existing_user:
                # Link existing user to this participant
                self.object.user = existing_user
                self.object.save()
                messages.info(
                    self.request,
                    f'ℹ️ Linked to existing user account for {email}.'
                )
            else:
                # Create new user account
                user = User.objects.create_user(
                    email=email,
                    password=None  # No password set yet - user will use password reset
                )
                
                # Link to participant
                self.object.user = user
                self.object.save()
                
                # Create the primary EmailAddress record for allauth
                EmailAddress.objects.create(
                    user=user,
                    email=email,
                    verified=False,  # Requires verification
                    primary=True
                )
                
        except Exception as e:
            messages.warning(
                self.request,
                f'⚠️ Participant created but failed to create user account: {str(e)}'
            )
    
    def form_invalid(self, form):
        """Handle form validation errors."""
        messages.error(
            self.request,
            '❌ Please correct the errors below and try again.'
        )
        return super().form_invalid(form)


class ParticipantUpdateView(AdminRequiredMixin, UpdateView):
    """
    View for editing existing participants.
    
    Features:
    - Pre-populated form with current data
    - Same styling and validation as create view
    - Success/error messaging
    - Redirect back to participant list
    """
    model = Participant
    form_class = ParticipantForm
    template_name = 'participants/form.html'
    success_url = reverse_lazy('participants:list')
    context_object_name = 'participant'
    
    def get_form_kwargs(self):
        """Pass the current user to the form for role-based field display."""
        kwargs = super().get_form_kwargs()
        kwargs['user'] = self.request.user
        return kwargs
    
    def form_valid(self, form):
        """Handle successful form submission with email sync."""
        try:
            # Check if email has changed
            old_email = self.object.email if self.object.pk else None
            new_email = form.cleaned_data.get('email')
            email_changed = old_email != new_email and new_email
            
            # Save the participant
            response = super().form_valid(form)
            
            # Handle email changes and User account creation/linking
            if email_changed:
                if self.object.user:
                    # Existing user - add new email
                    self.handle_email_change(old_email, new_email)
                elif new_email:
                    # No user yet but email provided - create User account
                    self.create_user_account(new_email)
            
            # Success message
            action_msg = f'✅ Participant "{self.object.name}" has been updated successfully!'
            if email_changed:
                action_msg += f' Email changed to {new_email}.'
                if self.object.user:
                    action_msg += ' User account is now linked.'
            
            messages.success(self.request, action_msg)
            return response
            
        except IntegrityError:
            messages.error(
                self.request,
                '❌ Unable to update participant. Please check for duplicate names or emails.'
            )
            return self.form_invalid(form)
    
    def create_user_account(self, email):
        """Create a User account and link it to the participant."""
        try:
            # Check if a user with this email already exists
            existing_user = User.objects.filter(email__iexact=email).first()
            
            if existing_user:
                # Link existing user to this participant
                self.object.user = existing_user
                self.object.save()
                messages.info(
                    self.request,
                    f'ℹ️ Linked to existing user account for {email}.'
                )
            else:
                # Create new user account
                user = User.objects.create_user(
                    email=email,
                    password=None  # No password set yet - user will use password reset
                )
                
                # Link to participant
                self.object.user = user
                self.object.save()
                
                # Create the primary EmailAddress record for allauth
                EmailAddress.objects.create(
                    user=user,
                    email=email,
                    verified=False,  # Requires verification
                    primary=True
                )
                
        except Exception as e:
            messages.warning(
                self.request,
                f'⚠️ Participant updated but failed to create user account: {str(e)}'
            )
    
    def handle_email_change(self, old_email, new_email):
        """Handle email changes by adding to User's email addresses."""
        try:
            # Check if the new email already exists in the system
            existing_email = EmailAddress.objects.filter(email__iexact=new_email).first()
            if existing_email and existing_email.user != self.object.user:
                messages.warning(
                    self.request,
                    f'⚠️ Email {new_email} is already in use by another account. '
                    f'Participant updated but email not added to user account.'
                )
                return
            
            # Add new email as unverified if it doesn't exist for this user
            if not existing_email:
                email_address = EmailAddress.objects.create(
                    user=self.object.user,
                    email=new_email,
                    verified=False,
                    primary=False
                )
                
                # Send verification email
                send_verification_email_to_address(self.request, email_address)
                
        except Exception as e:
            messages.warning(
                self.request,
                f'⚠️ Participant updated but failed to add email to user account: {str(e)}'
            )
    
    def form_invalid(self, form):
        """Handle form validation errors."""
        messages.error(
            self.request,
            '❌ Please correct the errors below and try again.'
        )
        return super().form_invalid(form)


class ParticipantDeleteView(AdminRequiredMixin, DeleteView):
    """
    View for deleting participants (admin only).
    
    Features:
    - Admin-only access with proper permission checking
    - Confirmation page with participant details
    - Handles cleanup of related User account if desired
    - Success message with proper redirect
    """
    model = Participant
    template_name = 'participants/delete.html'
    success_url = reverse_lazy('participants:list')
    context_object_name = 'participant'
    
    def delete(self, request, *args, **kwargs):
        """Handle the deletion with a success message."""
        self.object = self.get_object()
        participant_name = self.object.name or f"Participant {self.object.id}"
        
        # Get information for the success message
        has_user = bool(self.object.user)
        user_email = self.object.user.email if has_user else None
        question_count = self.object.question_set.count() if hasattr(self.object, 'question_set') else 0
        
        try:
            # Note: We're NOT deleting the User account, just unlinking
            # This preserves any User data but removes the Participant
            response = super().delete(request, *args, **kwargs)
            
            # Success message with details
            success_msg = f'✅ Participant "{participant_name}" has been deleted successfully.'
            if has_user:
                success_msg += f' Associated User account ({user_email}) remains active.'
            if question_count > 0:
                success_msg += f' Note: {question_count} question(s) were authored by this participant.'
            
            messages.success(request, success_msg)
            return response
            
        except Exception as e:
            messages.error(
                request,
                f'❌ Unable to delete participant "{participant_name}". '
                f'They may have associated questions or other data. Error: {str(e)}'
            )
            return redirect('participants:list')
    
    def get_context_data(self, **kwargs):
        """Add additional context for the confirmation page."""
        context = super().get_context_data(**kwargs)
        
        # Add related information for the confirmation page
        context['has_user'] = bool(self.object.user)
        context['user_email'] = self.object.user.email if self.object.user else None
        context['question_count'] = self.object.question_set.count() if hasattr(self.object, 'question_set') else 0
        
        return context


def check_email_exists(request):
    """
    AJAX endpoint to check if an email address already exists in the system.
    Used by the review invitation form to prevent duplicate accounts.
    """
    email = request.GET.get('email', '').strip()
    
    if not email:
        return JsonResponse({'exists': False})
    
    try:
        user = User.objects.get(email=email)
        return JsonResponse({
            'exists': True,
            'id': user.id,
            'name': user.get_full_name() or user.email
        })
    except User.DoesNotExist:
        return JsonResponse({'exists': False})
