"""
Database schemas for football_Ticket_Booking

This module contains all database schema classes for the football domain.
All classes are Pydantic BaseModel subclasses.

Generated by ClassDBAgent.
"""

import json
from datetime import date, datetime
from datetime import datetime
from datetime import datetime, date
from decimal import Decimal
from pathlib import Path
from pydantic import Field
from pydantic import Field, field_validator
from pydantic import PrivateAttr, Field
from pydantic import PrivateAttr, Field, field_validator
from pydantic import PrivateAttr, Field, validator
from scale_env.data_model.thread_safe_base import ThreadSafeBase, with_instance_key
from scale_env.environment.db import DB, DictAccessMixin as BaseModel
from typing import Dict, List, Any, Optional
from typing import Optional
from typing import Optional, List, Dict, Any

@with_instance_key("match_id")
class Match(BaseModel, ThreadSafeBase["Match"]):
    match_id: str = Field(default=..., description="Unique identifier for the match")
    home_team: str = Field(default=..., description="Name of the home team")
    away_team: str = Field(default=..., description="Name of the away team")
    match_date: datetime = Field(default=..., description="Date and time of the match")
    venue_name: str = Field(default=..., description="Name of the venue")
    competition: str = Field(default=..., description="Name of the competition")
    season: str = Field(default=..., description="Season information")
    is_postponed: bool = Field(default=False, description="Whether the match has been postponed")
    original_date: Optional[datetime] = Field(default=None, description="Original match date if rescheduled")
    postponement_reason: Optional[str] = Field(default=None, description="Reason for postponement if applicable")

    @validator('match_date', 'original_date', pre=True)
    def strip_and_parse_datetime(cls, v):
        if v is None:
            return v
        if isinstance(v, datetime):
            return v
        if isinstance(v, str):
            v = v.strip()
            if v == "":
                return None
            return datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
        return v

@with_instance_key("seat_id")
class Seat(BaseModel, ThreadSafeBase["Seat"]):
    seat_id: str = Field(default=..., description="Unique identifier for the seat")
    match_id: str = Field(default=..., description="Associated match identifier")
    venue_name: str = Field(default=..., description="Name of the venue")
    section: str = Field(default=..., description="Stadium section name")
    row: str = Field(default=..., description="Row identifier")
    seat_number: str = Field(default=..., description="Seat number within the row")
    category: str = Field(default=..., description="Seat category (VIP, Premium, Standard, etc.)")
    price: float = Field(default=..., description="Seat price")
    status: str = Field(default=..., description="Seat status (available, reserved, sold, used)")
    wheelchair_accessible: bool = Field(default=False, description="Whether seat is wheelchair accessible")
    companion_seat: bool = Field(default=False, description="Whether seat is designated as companion seat")
    elevator_access: bool = Field(default=False, description="Whether elevator access is available")

@with_instance_key("reservation_id")
class Reservation(BaseModel, ThreadSafeBase["Reservation"]):
    reservation_id: str = Field(default=..., description="Unique identifier for the reservation")
    match_id: str = Field(default=..., description="Associated match identifier")
    customer_id: str = Field(default=..., description="Customer who made the reservation")
    reservation_time: datetime = Field(default=..., description="Time when reservation was created")
    expiration_time: datetime = Field(default=..., description="Expiration time of the reservation")
    status: str = Field(default=..., description="Reservation status (active, expired, converted, released)")

@with_instance_key("reservation_seat_id")
class ReservationSeat(BaseModel, ThreadSafeBase["ReservationSeat"]):
    reservation_seat_id: str = Field(default=..., description="Unique identifier for the reservation-seat relationship")
    reservation_id: str = Field(default=..., description="Associated reservation identifier")
    seat_id: str = Field(default=..., description="Associated seat identifier")

@with_instance_key("booking_id")
class Booking(BaseModel, ThreadSafeBase["Booking"]):
    booking_id: str = Field(default=..., description="Unique identifier for the booking")
    booking_reference: str = Field(default=..., description="Human-readable booking reference code")
    match_id: str = Field(default=..., description="Associated match identifier")
    customer_id: str = Field(default=..., description="Customer who made the booking")
    reservation_id: Optional[str] = Field(default=None, description="Original reservation identifier")
    payment_id: str = Field(default=..., description="Associated payment transaction identifier")
    email: str = Field(default=..., description="Customer contact email")
    phone: Optional[str] = Field(default=None, description="Customer contact phone number")
    total_price: float = Field(default=..., description="Total price paid for the booking")
    booking_time: datetime = Field(default=..., description="Time when booking was created")
    status: str = Field(default=..., description="Current booking status (confirmed, cancelled, completed)")
    cancellation_time: Optional[datetime] = Field(default=None, description="Time when booking was cancelled")
    cancellation_reason: Optional[str] = Field(default=None, description="Reason for cancellation")
    refund_amount: Optional[float] = Field(default=None, description="Amount refunded if cancelled")

@with_instance_key("booking_seat_id")
class BookingSeat(BaseModel, ThreadSafeBase["BookingSeat"]):
    booking_seat_id: str = Field(..., description="Unique identifier for the booking-seat relationship")
    booking_id: str = Field(..., description="Associated booking identifier")
    seat_id: str = Field(..., description="Associated seat identifier")
    is_used: bool = Field(default=False, description="Whether ticket has been used for entry")
    entry_time: Optional[datetime] = Field(default=None, description="Time when ticket was used for entry")
    entry_gate: Optional[str] = Field(default=None, description="Gate number where entry occurred")
    qr_code_data: Optional[str] = Field(default=None, description="QR code data for ticket validation")

    @field_validator("entry_time", mode="before")
    @classmethod
    def strip_and_parse_datetime(cls, v):
        if v is None:
            return v
        if isinstance(v, datetime):
            return v
        if isinstance(v, str):
            v = v.strip()
            if not v:
                return None
            # Try parsing using common format
            try:
                return datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                # If it fails, let Pydantic try its own parsing
                return v
        return v

@with_instance_key("customer_id")
class Customer(BaseModel, ThreadSafeBase["Customer"]):
    customer_id: str = Field(default=..., description="Unique identifier for the customer")
    email: str = Field(default=..., description="Customer email address")
    phone: Optional[str] = Field(default=None, description="Customer phone number")
    is_season_ticket_holder: bool = Field(default=False, description="Whether customer is a season ticket holder")
    loyalty_tier: Optional[str] = Field(default=None, description="Customer loyalty tier (bronze, silver, gold, platinum)")
    total_loyalty_points: int = Field(default=0, description="Total accumulated loyalty points")
    age: Optional[int] = Field(default=None, description="Customer age in years")

@with_instance_key("payment_id")
class Payment(BaseModel, ThreadSafeBase["Payment"]):
    payment_id: str = Field(default=..., description="Unique identifier for the payment")
    booking_id: Optional[str] = Field(default=None, description="Associated booking identifier")
    customer_id: str = Field(default=..., description="Customer who made the payment")
    payment_method: str = Field(default=..., description="Payment method used (credit_card, debit_card, paypal, etc.)")
    amount: float = Field(default=..., description="Payment amount")
    payment_status: str = Field(default=..., description="Payment status (pending, completed, failed, refunded)")
    payment_time: datetime = Field(default=..., description="Time when payment was processed")
    transaction_reference: Optional[str] = Field(default=None, description="Payment gateway transaction reference")

@with_instance_key("discount_code")
class DiscountCode(BaseModel, ThreadSafeBase["DiscountCode"]):
    discount_code: str = Field(..., description="Discount code string")
    discount_type: str = Field(..., description="Type of discount (percentage or fixed_amount)")
    discount_value: float = Field(..., description="Discount value (percentage or amount)")
    expiration_date: date = Field(..., description="Expiration date of the discount code")
    is_active: bool = Field(default=True, description="Whether discount code is currently active")
    match_id: Optional[str] = Field(default=None, description="Specific match the code applies to (if applicable)")
    min_purchase_amount: Optional[float] = Field(default=None, description="Minimum purchase amount required")
    max_uses: Optional[int] = Field(default=None, description="Maximum number of times code can be used")
    times_used: int = Field(default=0, description="Number of times code has been used")

    @field_validator("expiration_date", mode="before")
    def parse_expiration_date(cls, v):
        if isinstance(v, date):
            return v
        if isinstance(v, datetime):
            return v.date()
        if isinstance(v, str):
            # Try to parse as datetime, then extract date
            try:
                dt = datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
                return dt.date()
            except ValueError:
                try:
                    # Fallback for ISO format or date-only string
                    dt = datetime.fromisoformat(v)
                    return dt.date()
                except Exception:
                    try:
                        d = date.fromisoformat(v)
                        return d
                    except Exception:
                        raise ValueError(f"Invalid expiration_date format: {v}")
        raise ValueError(f"Invalid expiration_date format: {v}")

@with_instance_key("transfer_id")
class TicketTransfer(BaseModel, ThreadSafeBase["TicketTransfer"]):
    transfer_id: str = Field(default=..., description="Unique identifier for the transfer")
    booking_id: str = Field(default=..., description="Original booking identifier")
    seat_id: str = Field(default=..., description="Seat being transferred")
    from_customer_id: str = Field(default=..., description="Original owner customer ID")
    to_customer_id: str = Field(default=..., description="New owner customer ID")
    transfer_time: datetime = Field(default=..., description="Time when transfer was completed")
    transfer_reason: Optional[str] = Field(default=None, description="Reason for transfer")
    new_booking_reference: str = Field(default=..., description="New booking reference for the recipient")

    @field_validator("transfer_time", mode="before")
    def parse_transfer_time(cls, v):
        if isinstance(v, str):
            v = v.strip()
            # Accept common datetime format
            try:
                return datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                pass
        return v

@with_instance_key("venue_name")
class Venue(BaseModel, ThreadSafeBase["Venue"]):
    venue_name: str = Field(default=..., description="Name of the venue")
    total_capacity: int = Field(default=..., description="Total seating capacity")
    address: Optional[str] = Field(default=None, description="Venue address")  # address is optional
    wifi_available: bool = Field(default=False, description="Whether WiFi is available")

@with_instance_key("section_id")
class VenueSection(BaseModel, ThreadSafeBase["VenueSection"]):
    section_id: str = Field(default=..., description="Unique identifier for the section")
    venue_name: str = Field(default=..., description="Associated venue name")
    section: str = Field(default=..., description="Name of the section")
    capacity: int = Field(default=..., description="Section capacity")

@with_instance_key("facility_id")
class VenueFacility(BaseModel, ThreadSafeBase["VenueFacility"]):
    facility_id: str = Field(default=..., description="Unique identifier for the facility")
    venue_name: str = Field(default=..., description="Associated venue name")
    facility_type: str = Field(default=..., description="Type of facility (parking, food_outlet, accessibility)")
    facility_name: str = Field(default=..., description="Name of the facility")
    capacity: Optional[int] = Field(default=None, description="Facility capacity (if applicable)") # capacity is optional
    price: Optional[float] = Field(default=None, description="Facility price (if applicable)") # price is optional

@with_instance_key("pricing_tier_id")
class PricingTier(BaseModel, ThreadSafeBase["PricingTier"]):
    pricing_tier_id: str = Field(default=..., description="Unique identifier for the pricing tier")
    match_id: str = Field(default=..., description="Associated match identifier")
    tier_name: str = Field(default=..., description="Name of the pricing tier")
    base_price: Decimal = Field(default=..., description="Base price for the tier")
    member_price: Optional[Decimal] = Field(default=None, description="Price for members")
    child_price: Optional[Decimal] = Field(default=None, description="Price for children")
    senior_price: Optional[Decimal] = Field(default=None, description="Price for seniors")

@with_instance_key("season_ticket_id")
class SeasonTicket(BaseModel, ThreadSafeBase["SeasonTicket"]):
    season_ticket_id: str = Field(default=..., description="Unique identifier for the season ticket")
    customer_id: str = Field(default=..., description="Associated customer identifier")
    season: str = Field(default=..., description="Season identifier")
    venue_name: str = Field(default=..., description="Venue name")
    section: str = Field(default=..., description="Assigned section")
    row: str = Field(default=..., description="Assigned row")
    seat_number: str = Field(default=..., description="Assigned seat number")
    priority_window_start: Optional[datetime] = Field(default=None, description="Start time of priority booking window")  # priority_window_start is optional

@with_instance_key("waitlist_id")
class Waitlist(BaseModel, ThreadSafeBase["Waitlist"]):
    waitlist_id: str = Field(..., description="Unique identifier for the waitlist entry")
    customer_id: str = Field(..., description="Customer on the waitlist")  # references customer.customer_id
    match_id: str = Field(..., description="Match identifier")  # references match.match_id
    ticket_quantity: int = Field(..., description="Number of tickets desired")
    preferred_category: Optional[str] = Field(default=None, description="Preferred seat category")
    position: int = Field(..., description="Position in the waitlist")
    joined_time: datetime = Field(..., description="Time when customer joined waitlist")
    status: str = Field(..., description="Waitlist status (active, notified, converted, expired)")
    notification_sent: bool = Field(default=False, description="Whether availability notification has been sent")

@with_instance_key("resale_listing_id")

class ResaleListing(BaseModel, ThreadSafeBase["ResaleListing"]):
    resale_listing_id: str = Field(default=..., description="Unique identifier for the resale listing")
    booking_id: str = Field(default=..., description="Original booking identifier")
    seat_id: str = Field(default=..., description="Seat being listed for resale")
    seller_customer_id: str = Field(default=..., description="Customer ID of the seller")
    resale_price: float = Field(default=..., description="Price listed for resale")
    listed_time: datetime = Field(default=..., description="Time when ticket was listed")
    listing_status: str = Field(default=..., description="Current status of the listing (active, sold, expired, withdrawn)")
    sold_time: Optional[datetime] = Field(default=None, description="Time when ticket was sold")
    buyer_customer_id: Optional[str] = Field(default=None, description="Customer ID of the buyer (if sold)")
    seller_payout_amount: Optional[float] = Field(default=None, description="Amount to be paid to seller after fees")

@with_instance_key("message_id")
class CommunicationLog(BaseModel, ThreadSafeBase["CommunicationLog"]):
    message_id: str = Field(default=..., description="Unique identifier for the message")
    customer_id: str = Field(default=..., description="Recipient customer identifier")
    booking_id: Optional[str] = Field(default=None, description="Associated booking identifier (if applicable)")
    communication_type: str = Field(default=..., description="Type of communication (confirmation, reminder, notification)")
    channel: str = Field(default=..., description="Communication channel (email, sms, push)")
    recipient_address: str = Field(default=..., description="Email address or phone number")
    sent_time: datetime = Field(default=..., description="Time when message was sent")
    status: str = Field(default=..., description="Delivery status (sent, delivered, failed)")

    @field_validator('sent_time', mode='before')
    @classmethod
    def strip_and_parse_datetime(cls, v):
        if isinstance(v, str):
            v = v.strip()
            try:
                # Try parsing with standard format
                return datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                pass
        return v

class FootballDB(DB):
    """Database containing all football_Ticket_Booking-related data"""
    match: Optional[Dict[str, Match]] = Field(
        default=None,
        description="Schema Match"
    )
    seat: Optional[Dict[str, Seat]] = Field(
        default=None,
        description="Schema Seat"
    )
    reservation: Optional[Dict[str, Reservation]] = Field(
        default=None,
        description="Schema Reservation"
    )
    reservation_seat: Optional[Dict[str, ReservationSeat]] = Field(
        default=None,
        description="Schema ReservationSeat"
    )
    booking: Optional[Dict[str, Booking]] = Field(
        default=None,
        description="Schema Booking"
    )
    booking_seat: Optional[Dict[str, BookingSeat]] = Field(
        default=None,
        description="Schema BookingSeat"
    )
    customer: Optional[Dict[str, Customer]] = Field(
        default=None,
        description="Schema Customer"
    )
    payment: Optional[Dict[str, Payment]] = Field(
        default=None,
        description="Schema Payment"
    )
    discount_code: Optional[Dict[str, DiscountCode]] = Field(
        default=None,
        description="Schema DiscountCode"
    )
    ticket_transfer: Optional[Dict[str, TicketTransfer]] = Field(
        default=None,
        description="Schema TicketTransfer"
    )
    venue: Optional[Dict[str, Venue]] = Field(
        default=None,
        description="Schema Venue"
    )
    venue_section: Optional[Dict[str, VenueSection]] = Field(
        default=None,
        description="Schema VenueSection"
    )
    venue_facility: Optional[Dict[str, VenueFacility]] = Field(
        default=None,
        description="Schema VenueFacility"
    )
    pricing_tier: Optional[Dict[str, PricingTier]] = Field(
        default=None,
        description="Schema PricingTier"
    )
    season_ticket: Optional[Dict[str, SeasonTicket]] = Field(
        default=None,
        description="Schema SeasonTicket"
    )
    waitlist: Optional[Dict[str, Waitlist]] = Field(
        default=None,
        description="Schema Waitlist"
    )
    resale_listing: Optional[Dict[str, ResaleListing]] = Field(
        default=None,
        description="Schema ResaleListing"
    )
    communication_log: Optional[Dict[str, CommunicationLog]] = Field(
        default=None,
        description="Schema CommunicationLog"
    )