"""
Database schemas for entertainment_media_query

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

Generated by ClassDBAgent.
"""

import json
from datetime import datetime
from datetime import datetime, date
from pathlib import Path
from pydantic import Field
from pydantic import PrivateAttr, Field
from pydantic import PrivateAttr, Field, 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 Dict, List, Any, Optional, Literal

@with_instance_key("movie_id")
class Movie(BaseModel, ThreadSafeBase["Movie"]):      
    movie_id: str = Field(default=..., description="Unique identifier for the movie (e.g., mov_12345).")
    title: str = Field(default=..., description="Official title of the movie.")
    plot_synopsis: Optional[str] = Field(default=None, description="Detailed plot summary of the movie.")
    release_date: Optional[datetime] = Field(default=None, description="Official release date in yyyy-mm-dd format.")
    runtime: Optional[int] = Field(default=None, description="Duration of the movie in minutes.")
    average_rating: Optional[float] = Field(default=None, description="Aggregate rating score out of 10.")
    vote_count: Optional[int] = Field(default=0, description="Total number of votes for the rating.")
    content_rating: Optional[Literal["G", "PG", "PG-13", "R", "NC-17", "NR"]] = Field(default=None, description="Content rating classification.")
    added_date: datetime = Field(default=..., description="The date and time when the movie was added to the database.")

@with_instance_key("series_id")
class TvSeries(BaseModel, ThreadSafeBase["TvSeries"]):      
    series_id: str = Field(default=..., description="Unique identifier for the TV series (e.g., tv_67890).")
    title: str = Field(default=..., description="Official title of the TV series.")
    plot_synopsis: Optional[str] = Field(default=None, description="Detailed plot summary.")
    release_date: Optional[datetime] = Field(default=None, description="Date the first episode aired.")
    last_air_date: Optional[datetime] = Field(default=None, description="Date the last episode aired.")
    number_of_seasons: Optional[int] = Field(default=None, description="Total number of seasons produced.")
    episode_count: Optional[int] = Field(default=None, description="Total number of episodes produced.")
    average_rating: Optional[float] = Field(default=None, description="Aggregate rating score.")
    vote_count: Optional[int] = Field(default=None, description="Total number of votes for the rating.")
    status: Optional[Literal["returning_series", "ended", "in_production", "canceled", "planned"]] = Field(default=None, description="Current production status of the series.")
    content_rating: Optional[Literal["TV-Y", "TV-Y7", "TV-G", "TV-PG", "TV-14", "TV-MA", "NR"]] = Field(default=None, description="Content rating classification for TV.")
    added_date: datetime = Field(default=..., description="The date and time when the series was added to the database.")

@with_instance_key("anime_id")
class Anime(BaseModel, ThreadSafeBase["Anime"]):      
    anime_id: str = Field(default=..., description="Unique identifier for the anime (e.g., ani_11111).")
    title: str = Field(default=..., description="Official title of the anime.")
    plot_synopsis: Optional[str] = Field(default=None, description="Detailed plot summary.")
    release_date: Optional[datetime] = Field(default=None, description="Date the first episode aired.")
    last_air_date: Optional[datetime] = Field(default=None, description="Date the last episode aired.")
    episode_count: Optional[int] = Field(default=None, description="Total number of episodes.")
    average_rating: Optional[float] = Field(default=None, description="Aggregate rating score.")
    vote_count: Optional[int] = Field(default=None, description="Total number of votes.")
    company_name: Optional[str] = Field(default=None, description="Animation studio that produced the anime.")
    status: Optional[Literal["currently_airing", "finished_airing", "not_yet_aired"]] = Field(default=None, description="Current broadcast status.")
    content_rating: Optional[Literal["G", "PG", "PG-13", "R", "R+", "Rx"]] = Field(default=None, description="Content rating classification for anime.")
    added_date: datetime = Field(default=..., description="The date and time when the anime was added.")

    @field_validator("release_date", "last_air_date", "added_date", mode="before")
    @classmethod
    def strip_datetime_whitespace(cls, v: Any) -> Any:
        """
        Fixes validation errors where date strings contain trailing whitespace.
        Pydantic's default datetime parser in strict mode rejects extra characters.
        """
        if isinstance(v, str):
            return v.strip()
        return v

@with_instance_key("season_id")
class Season(BaseModel, ThreadSafeBase["Season"]):      
    """
    Stores information about seasons of a TV series.
    """
    season_id: str = Field(default=..., description="Unique identifier for the season.")
    series_id: str = Field(default=..., description="Reference to the TV series.")
    season_number: int = Field(default=..., description="The number of the season (e.g., 1, 2).")
    episode_count: Optional[int] = Field(default=None, description="Number of episodes in this season.")
    release_date: Optional[datetime] = Field(default=None, description="Premiere date of the season.")

@with_instance_key("episode_id")
class Episode(BaseModel, ThreadSafeBase["Episode"]):      
    """
    Stores information about individual episodes for TV series or anime.
    """
    episode_id: str = Field(..., description="Unique identifier for the episode.")
    content_id: str = Field(..., description="Reference to the parent series_id or anime_id.")
    season_number: Optional[int] = Field(default=None, description="Season number (primarily for TV series).")
    episode_number: int = Field(..., description="The number of the episode within the season or series.")
    title: Optional[str] = Field(default=None, description="Title of the episode.")
    release_date: Optional[datetime] = Field(default=None, description="Date the episode aired.")
    runtime: Optional[int] = Field(default=None, description="Duration of the episode in minutes.")

@with_instance_key("person_id")
class Person(BaseModel, ThreadSafeBase["Person"]):      
    """
    Stores information about individuals involved in productions (actors, directors, staff).
    """
    person_id: str = Field(default=..., description="Unique identifier for the person.")
    name: str = Field(default=..., description="Full name of the person.")

@with_instance_key("cast_id")
class CastMember(BaseModel, ThreadSafeBase["CastMember"]):      
    cast_id: str = Field(default=..., description="Unique identifier for the cast entry.")
    content_id: str = Field(default=..., description="Reference to the movie, TV series, or anime.")
    person_id: str = Field(default=..., description="Reference to the person.")
    character_name: Optional[str] = Field(default=None, description="Name of the character played.")
    role: Literal["actor", "voice_actor"] = Field(default=..., description="Type of acting role.")
    billing_order: Optional[int] = Field(default=None, description="Order of appearance in credits (for movies).")
    episode_count: Optional[int] = Field(default=None, description="Number of episodes appeared in (for TV).")
    language: Optional[str] = Field(default=None, description="Language of the voice acting (for anime).")

@with_instance_key("crew_id")
class CrewMember(BaseModel, ThreadSafeBase["CrewMember"]):      
    crew_id: str = Field(default=..., description="Unique identifier for the crew entry.")
    content_id: str = Field(default=..., description="Reference to the movie, TV series, or anime.")
    person_id: str = Field(default=..., description="Reference to the person.")
    role: str = Field(default=..., description="Role name (e.g., Director, Creator, Staff).")
    department: Optional[str] = Field(default=None, description="Department name (e.g., Directing, Writing).")

@with_instance_key("genre_id")
class Genre(BaseModel, ThreadSafeBase["Genre"]):      
    """Master list of available genres."""
    genre_id: str = Field(default=..., description="Unique identifier for the genre.")
    genre_name: str = Field(default=..., description="Name of the genre.")

@with_instance_key("id")
class ContentGenre(BaseModel, ThreadSafeBase["ContentGenre"]):      
    """
    Links movies, TV series, or anime to their respective genres.
    """
    id: str = Field(default=..., description="Unique identifier for the link.")
    content_id: str = Field(default=..., description="Reference to the content.")
    genre_id: str = Field(default=..., description="Reference to the genre.")

@with_instance_key("award_id")
class Award(BaseModel, ThreadSafeBase["Award"]):      
    award_id: str = Field(default=..., description="Unique identifier for the award entry.")
    content_id: str = Field(default=..., description="Reference to the content.")
    award_name: str = Field(default=..., description="Name of the award (e.g., Academy Awards).")
    category: Optional[str] = Field(default=None, description="Category of the award.")
    result: Literal["Won", "Nominated"] = Field(default=..., description="Outcome of the award.")
    year: Optional[int] = Field(default=None, description="Year the award was presented.")

@with_instance_key("company_id")
class ProductionCompany(BaseModel, ThreadSafeBase["ProductionCompany"]):      
    """Stores information about companies involved in production."""
    company_id: str = Field(default=..., description="Unique identifier for the company.")
    company_name: str = Field(default=..., description="Official name of the company.")
    country: Optional[str] = Field(default=None, description="Country where the company is based.")

@with_instance_key("id")
class ContentProductionCompany(BaseModel, ThreadSafeBase["ContentProductionCompany"]):      
    id: str = Field(default=..., description="Unique identifier for the link.")
    content_id: str = Field(default=..., description="Reference to the content.")
    company_id: str = Field(default=..., description="Reference to the company.")

@with_instance_key("asset_id")
class MediaAsset(BaseModel, ThreadSafeBase["MediaAsset"]):      
    asset_id: str = Field(default=..., description="Unique identifier for the asset.")
    content_id: str = Field(default=..., description="Reference to the content.")
    asset_type: Literal["trailer", "poster", "backdrop", "still", "character_art", "all"] = Field(default=..., description="Type of media asset.")
    url: str = Field(default=..., description="URL to the media file.")
    title: Optional[str] = Field(default=None, description="Title of the asset (for trailers).")
    runtime: Optional[int] = Field(default=None, description="Duration in seconds (for trailers).")
    release_date: Optional[datetime] = Field(default=None, description="Release date of the asset.")
    width: Optional[int] = Field(default=None, description="Width in pixels (for images).")
    height: Optional[int] = Field(default=None, description="Height in pixels (for images).")

    @field_validator("release_date", mode="before")
    @classmethod
    def strip_release_date(cls, v: Any) -> Any:
        """
        Validator to handle trailing whitespaces in date strings which cause
        Pydantic's default datetime parser to fail.
        """
        if isinstance(v, str):
            return v.strip()
        return v

@with_instance_key("alt_title_id")
class AlternativeTitle(BaseModel, ThreadSafeBase["AlternativeTitle"]):      
    alt_title_id: str = Field(default=..., description="Unique identifier for the alternative title.")
    content_id: str = Field(default=..., description="Reference to the content.")
    title: str = Field(default=..., description="The alternative title.")
    language: Optional[str] = Field(default=None, description="Language of the title.")
    region: Optional[str] = Field(default=None, description="Region code (e.g., US, JP).")

@with_instance_key("popularity_id")
class ContentPopularity(BaseModel, ThreadSafeBase["ContentPopularity"]):      
    popularity_id: str = Field(default=..., description="Unique identifier for the popularity entry.")
    content_id: str = Field(default=..., description="Reference to the content.")
    region: str = Field(default=..., description="ISO country code or region identifier.")
    popularity_score: float = Field(default=..., description="Metric representing popularity in the region.")

@with_instance_key("id")
class ContentLanguage(BaseModel, ThreadSafeBase["ContentLanguage"]):      
    id: str = Field(default=..., description="Unique identifier for the language entry.")
    content_id: str = Field(default=..., description="Reference to the content.")
    language: str = Field(default=..., description="Name of the language.")
    language_type: Literal["audio", "subtitle"] = Field(default=..., description="Type of language track.")

class EntertainmentMediaQueryDB(DB):
    """Database containing all entertainment_media_query-related data"""
    movie: Optional[Dict[str, Movie]] = Field(
        default=None,
        description="Schema Movie"
    )
    tv_series: Optional[Dict[str, TvSeries]] = Field(
        default=None,
        description="Schema TvSeries"
    )
    anime: Optional[Dict[str, Anime]] = Field(
        default=None,
        description="Schema Anime"
    )
    season: Optional[Dict[str, Season]] = Field(
        default=None,
        description="Stores information about seasons of a TV series."
    )
    episode: Optional[Dict[str, Episode]] = Field(
        default=None,
        description="Stores information about individual episodes for TV series or anime."
    )
    person: Optional[Dict[str, Person]] = Field(
        default=None,
        description="Stores information about individuals involved in productions (actors, directors, staff)."
    )
    cast_member: Optional[Dict[str, CastMember]] = Field(
        default=None,
        description="Schema CastMember"
    )
    crew_member: Optional[Dict[str, CrewMember]] = Field(
        default=None,
        description="Schema CrewMember"
    )
    genre: Optional[Dict[str, Genre]] = Field(
        default=None,
        description="Master list of available genres."
    )
    content_genre: Optional[Dict[str, ContentGenre]] = Field(
        default=None,
        description="Links movies, TV series, or anime to their respective genres."
    )
    award: Optional[Dict[str, Award]] = Field(
        default=None,
        description="Schema Award"
    )
    production_company: Optional[Dict[str, ProductionCompany]] = Field(
        default=None,
        description="Stores information about companies involved in production."
    )
    content_production_company: Optional[Dict[str, ContentProductionCompany]] = Field(
        default=None,
        description="Schema ContentProductionCompany"
    )
    media_asset: Optional[Dict[str, MediaAsset]] = Field(
        default=None,
        description="Schema MediaAsset"
    )
    alternative_title: Optional[Dict[str, AlternativeTitle]] = Field(
        default=None,
        description="Schema AlternativeTitle"
    )
    content_popularity: Optional[Dict[str, ContentPopularity]] = Field(
        default=None,
        description="Schema ContentPopularity"
    )
    content_language: Optional[Dict[str, ContentLanguage]] = Field(
        default=None,
        description="Schema ContentLanguage"
    )