from typing import List, Dict, Optional, Any, Literal
from scale_env.environment.toolkit import ToolKitBase, ToolType, is_tool
from .database import *
from thefuzz import fuzz, process

"""Tools for entertainment_media_query."""

class EntertainmentMediaQueryTools(ToolKitBase):
    """All tools for entertainment_media_query."""
    
    db: EntertainmentMediaQueryDB
    
    def __init__(self, db: EntertainmentMediaQueryDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def search_movies_by_keyword(self, keyword: str, limit: int = None):
        """
        Search for movies using a keyword that matches title, plot, or other text fields.
        Returns a list of matching movie identifiers and basic information.
        """
        # Validate the keyword input as per pre-condition
        if not keyword or not isinstance(keyword, str):
            raise ValueError("Keyword must be a non-empty string")

        # Access the movie table from the database
        # getattr is used to safely retrieve the 'movie' dictionary from the db object
        movie_table = getattr(self.db, 'movie', {}) or {}

        if not movie_table:
            return {"results": []}

        # List to store potential matches with their similarity scores
        scored_matches = []

        # Iterate through all movies in the database to perform fuzzy matching
        for movie in movie_table.values():
            # Case-insensitive matching for better recall
            search_keyword = keyword.lower()

            # Calculate similarity for the title
            title_score = fuzz.partial_ratio(search_keyword, movie.title.lower())

            # Calculate similarity for the plot synopsis if it exists
            plot_score = 0
            if movie.plot_synopsis:
                plot_score = fuzz.partial_ratio(search_keyword, movie.plot_synopsis.lower())

            # Determine the best score for this movie across searchable fields
            best_score = max(title_score, plot_score)

            # Use a similarity threshold (60) to filter out irrelevant results
            # A threshold of 60 is generally effective for partial keyword matches
            if best_score >= 60:
                scored_matches.append((best_score, movie))

        # Sort the matches by their similarity score in descending order (most relevant first)
        scored_matches.sort(key=lambda x: x[0], reverse=True)

        # Apply the limit if provided
        if limit is not None and limit > 0:
            scored_matches = scored_matches[:limit]

        # Format the final output according to the returns schema
        results = []
        for score, movie in scored_matches:
            # Extract the release year from the datetime object
            release_year = None
            if movie.release_date:
                release_year = movie.release_date.year

            results.append({
                "movie_id": movie.movie_id,
                "title": movie.title,
                "release_year": release_year
            })

        return {"results": results}

    @is_tool()
    def filter_movies_by_runtime_range(self, min_runtime_minutes: int, max_runtime_minutes: int, limit: int = None):
        """
        Filter and retrieve movies with runtime duration within a specified range in minutes.
        """
        # 1. Validate the input parameters based on the pre-conditions
        # Both must be positive integers
        if min_runtime_minutes <= 0 or max_runtime_minutes <= 0:
            raise ValueError("min_runtime_minutes and max_runtime_minutes must be positive integers (greater than 0).")

        # Minimum runtime must be less than or equal to maximum runtime
        if min_runtime_minutes > max_runtime_minutes:
            raise ValueError("min_runtime_minutes must be less than or equal to max_runtime_minutes.")

        # 2. Access the database and the movie table
        # According to the instructions, self.db is the database instance
        movie_table = getattr(self.db, "movie", None)

        # If the table is empty or doesn't exist, return an empty list
        if not movie_table:
            return {"results": []}

        # 3. Iterate through movies and filter by runtime
        filtered_movies = []

        # movie_table is a Dict[str, Movie] where key is movie_id
        for movie in movie_table.values():
            # The Movie schema has a 'runtime' field (Optional[int])
            if movie.runtime is not None:
                # Check if it falls within the inclusive range
                if min_runtime_minutes <= movie.runtime <= max_runtime_minutes:
                    filtered_movies.append({
                        "movie_id": movie.movie_id,
                        "title": movie.title,
                        "runtime_minutes": movie.runtime
                    })

        # 4. Sort results if necessary (optional, but good for consistency)
        # The requirement doesn't specify sorting, but we'll keep the order from the DB or insertion order

        # 5. Apply the limit if provided
        if limit is not None and limit > 0:
            filtered_movies = filtered_movies[:limit]

        # 6. Return the results in the specified format
        return {"results": filtered_movies}

    @is_tool()
    def search_movies_by_director(self, director_name: str, limit: int = None):
        """
        Search for movies directed by a specific director by director name.
        Uses fuzzy matching for director names to handle partial or slightly different name variations.
        """
        # Import necessary modules inside the method as per requirements
        from datetime import datetime

        # Pre-condition validation: director_name must be a non-empty string
        if not director_name or not isinstance(director_name, str):
            raise ValueError("Director name must be a non-empty string.")

        # Access database tables
        movie_table = getattr(self.db, 'movie', {}) or {}
        person_table = getattr(self.db, 'person', {}) or {}
        crew_member_table = getattr(self.db, 'crew_member', {}) or {}

        # 1. Find the person(s) matching the director_name using fuzzy matching
        # We use person names as choices for fuzzy matching
        person_map = {p.person_id: p.name for p in person_table.values()}

        # Use process.extract to find matches. We'll use a threshold to ensure quality.
        # process.extract with a dict returns a list of (value, score, key)
        matches = process.extract(director_name, person_map, limit=None)

        # We filter matches with a score threshold (e.g., 85) to allow for partial names or small typos
        # but still keep the results relevant.
        matched_person_ids = [person_id for name, score, person_id in matches if score >= 85]

        if not matched_person_ids:
            return {"movies": []}

        # 2. Find crew member entries for these persons where the role is 'Director'
        # The schema specifies role as "Director" (e.g., in the CrewMember description)
        movie_ids = []
        for crew in crew_member_table.values():
            if crew.person_id in matched_person_ids and crew.role.lower() == "director":
                movie_ids.append(crew.content_id)

        # Remove duplicates if any (a director might have multiple entries for the same movie, though unlikely)
        movie_ids = list(set(movie_ids))

        # 3. Retrieve movie details and format the result
        results = []
        for m_id in movie_ids:
            movie = movie_table.get(m_id)
            if movie:
                # Extract release year from the release_date datetime object
                release_year = None
                if movie.release_date and isinstance(movie.release_date, datetime):
                    release_year = movie.release_date.year

                results.append({
                    "movie_id": movie.movie_id,
                    "title": movie.title,
                    "release_year": release_year
                })

        # Sort movies by release year descending for better usability (optional but good practice)
        results.sort(key=lambda x: (x['release_year'] if x['release_year'] is not None else 0), reverse=True)

        # 4. Apply limit if provided
        if limit is not None and isinstance(limit, int) and limit > 0:
            results = results[:limit]

        return {"movies": results}

    @is_tool()
    def filter_anime_by_content_rating(self, content_rating: Literal["G", "PG", "PG-13", "R", "R+", "Rx"], limit: int = None):
        """
        Filter and retrieve anime by content rating classification (e.g., G, PG, PG-13, R, R+).

        Args:
            content_rating (Literal["G", "PG", "PG-13", "R", "R+", "Rx"]): Content rating classification to filter by.
            limit (int, optional): Maximum number of results to return.

        Returns:
            dict: A dictionary containing the list of filtered anime results.
        """
        # Safety protection for enum parameter to ensure it matches the allowed values and handles the 'raises' requirement
        valid_ratings = ["G", "PG", "PG-13", "R", "R+", "Rx"]
        if content_rating not in valid_ratings:
            raise ValueError(f"Invalid content_rating: {content_rating}. Must be one of {valid_ratings}")

        # Access the database through the self.db attribute
        db = self.db
        # Note: Accessing the 'anime' table as it is necessary for the tool's core logic.
        anime_table = getattr(db, "anime", {})

        if not anime_table:
            return {"results": []}

        filtered_results = []

        # Iterate through all anime entries in the database
        for anime_obj in anime_table.values():
            # Check if the content rating matches the specified filter
            # Exact matching is appropriate here as content_rating is a classification code/enum, 
            # not a natural language text field like title or description.
            if anime_obj.content_rating == content_rating:
                filtered_results.append({
                    "anime_id": anime_obj.anime_id,
                    "title": anime_obj.title,
                    "content_rating": anime_obj.content_rating
                })

        # Apply limit if provided and valid
        if limit is not None and limit > 0:
            filtered_results = filtered_results[:limit]

        return {"results": filtered_results}

    @is_tool()
    def get_popular_anime_by_region(self, region_code: str, limit: int = None):
        """
        Retrieve a list of popular anime in a specific geographic region based on viewership or popularity metrics.

        Args:
            region_code (str): ISO country code or region identifier.
            limit (int, optional): Maximum number of results to return.

        Returns:
            dict: A dictionary containing a list of popular anime in the region.

        Raises:
            ValueError: If region_code is invalid or if limit is not a non-negative integer.
        """
        # Parameter validation
        if not isinstance(region_code, str) or not region_code.strip():
            raise ValueError("region_code must be a non-empty string.")

        if limit is not None:
            if not isinstance(limit, int):
                raise ValueError("limit must be an integer.")
            if limit < 0:
                raise ValueError("limit must be a non-negative integer.")

        # Access the database tables safely
        # Using getattr as recommended, with a fallback to an empty dictionary if the table is None
        popularity_table = getattr(self.db, "content_popularity", None) or {}
        anime_table = getattr(self.db, "anime", None) or {}

        # Normalization for exact matching (since region_code is an identifier/code)
        # We strip whitespace and convert to uppercase for robustness
        target_region = region_code.strip().upper()

        # Filter popularity records based on the region_code
        regional_entries = [
            entry for entry in popularity_table.values()
            if entry.region.strip().upper() == target_region
        ]

        # Sort the filtered entries by popularity_score in descending order (highest score first)
        regional_entries.sort(key=lambda x: x.popularity_score, reverse=True)

        # Map the popularity entries to the required return format by joining with the Anime table
        # We perform the join before applying the limit to ensure we return the requested number of valid anime
        popular_anime_list = []
        for entry in regional_entries:
            # Find the corresponding anime details using content_id (which maps to anime_id)
            anime_info = anime_table.get(entry.content_id)

            # Only include if the anime record exists in the anime table
            if anime_info:
                popular_anime_list.append({
                    "anime_id": entry.content_id,
                    "title": anime_info.title,
                    "popularity_score": entry.popularity_score
                })

        # Apply the limit if provided
        if limit is not None:
            popular_anime_list = popular_anime_list[:limit]

        # Return the result in the format specified by the tool schema
        return {"popular_anime": popular_anime_list}

    @is_tool()
    def filter_anime_by_genre(self, genres: List[Literal["Action", "Adventure", "Comedy", "Drama", "Ecchi", "Fantasy", "Horror", "Mecha", "Music", "Mystery", "Psychological", "Romance", "Sci-Fi", "Slice of Life", "Sports", "Supernatural", "Thriller"]], match_all: bool = False, limit: int = 20):
        """
        Filter and retrieve anime that belong to specified genres. Supports multiple genre filtering.
        """
        # Check if genres list is provided as per schema requirement and raises field
        if not genres:
            raise ValueError("The 'genres' parameter must be a non-empty list.")

        # Access the database tables
        anime_db = getattr(self.db, "anime", {})
        genre_db = getattr(self.db, "genre", {})
        content_genre_db = getattr(self.db, "content_genre", {})

        if not anime_db or not genre_db or not content_genre_db:
            return {"results": []}

        # 1. Map requested genre names to database genre IDs using fuzzy matching for robustness
        # Create a mapping of genre_name to genre_id from the database
        db_genre_names = {g.genre_name: g.genre_id for g in genre_db.values()}

        target_genre_ids = set()
        for genre_name in genres:
            # Use fuzzy matching to find the best match for the input genre name in the DB
            # process.extractOne is imported at the file level
            match_data = process.extractOne(genre_name, db_genre_names.keys())
            if match_data:
                match_str, score = match_data[0], match_data[1]
                if score >= 80:  # Threshold for a confident match
                    target_genre_ids.add(db_genre_names[match_str])

        # If no valid genres are found after fuzzy matching, return empty results
        if not target_genre_ids:
            return {"results": []}

        # 2. Map anime_ids to their sets of genre_ids and genre_names
        # This helps in filtering and also in constructing the final return object
        anime_to_genre_ids = {}
        anime_to_genre_names = {}

        # Pre-map genre_id to name for quick lookup
        id_to_name = {g.genre_id: g.genre_name for g in genre_db.values()}

        for cg in content_genre_db.values():
            aid = cg.content_id
            gid = cg.genre_id

            if aid not in anime_to_genre_ids:
                anime_to_genre_ids[aid] = set()
                anime_to_genre_names[aid] = []

            anime_to_genre_ids[aid].add(gid)
            if gid in id_to_name:
                anime_to_genre_names[aid].append(id_to_name[gid])

        # 3. Filter anime based on genre matching logic
        matched_anime_ids = []
        for anime_id, anime_genres in anime_to_genre_ids.items():
            if match_all:
                # Must contain all specified genres
                if target_genre_ids.issubset(anime_genres):
                    matched_anime_ids.append(anime_id)
            else:
                # Must contain at least one of the specified genres
                if not target_genre_ids.isdisjoint(anime_genres):
                    matched_anime_ids.append(anime_id)

        # 4. Construct the results list
        results = []
        for aid in matched_anime_ids:
            # Check if the anime exists in the anime table
            anime_obj = anime_db.get(aid)
            if anime_obj:
                results.append({
                    "anime_id": anime_obj.anime_id,
                    "title": anime_obj.title,
                    "genres": anime_to_genre_names.get(aid, [])
                })

        # Apply the limit if specified
        if limit and limit > 0:
            results = results[:limit]

        return {"results": results}

    @is_tool()
    def get_recently_added_tv_series(self, limit: int, days_back: int = None):
        """
        Retrieve a list of TV series that were recently added to the database, 
        sorted by addition date in descending order.
        """
        from datetime import datetime, timedelta

        # Input validation: limit must be a positive integer
        if limit <= 0:
            raise ValueError("The limit must be a positive integer.")

        # Access the tv_series table from the database
        # getattr is used to safely access the table, defaulting to an empty dict if not found
        tv_series_table = getattr(self.db, "tv_series", {})
        if not tv_series_table:
            return {"recently_added": []}

        # Extract series objects into a list
        series_list = list(tv_series_table.values())

        # Calculate the cutoff date if days_back is provided
        cutoff_date = None
        if days_back is not None:
            if days_back < 0:
                raise ValueError("days_back must be a non-negative integer.")
            cutoff_date = datetime.now() - timedelta(days=days_back)

        # Filter by days_back if applicable
        filtered_series = []
        for series in series_list:
            # added_date is required in the schema and is a datetime object
            if cutoff_date:
                if series.added_date >= cutoff_date:
                    filtered_series.append(series)
            else:
                filtered_series.append(series)

        # Sort the series by added_date in descending order
        # Most recently added series first
        sorted_series = sorted(
            filtered_series, 
            key=lambda x: x.added_date, 
            reverse=True
        )

        # Apply the limit to the result set
        paged_series = sorted_series[:limit]

        # Format the output according to the tool schema requirements
        # Dates must be converted to "yyyy-mm-dd HH:MM:SS" strings
        recently_added = []
        for series in paged_series:
            formatted_date = series.added_date.strftime("%Y-%m-%d %H:%M:%S")
            recently_added.append({
                "series_id": series.series_id,
                "title": series.title,
                "added_date": formatted_date
            })

        return {"recently_added": recently_added}

    @is_tool()
    def search_anime_by_keyword(self, keyword: str, limit: int = 10):
        """
        Searches for anime in the database using a keyword. The search matches against
        the title and plot synopsis using fuzzy matching to ensure robustness against
        minor typos or partial matches.
        """
        # Pre-condition check: Keyword must be a non-empty string
        if not keyword or not isinstance(keyword, str) or not keyword.strip():
            raise ValueError("Keyword must be a non-empty string")

        # Access the anime table from the database
        anime_table = getattr(self.db, "anime", {})
        if not anime_table:
            return {"results": []}

        scored_results = []
        search_keyword = keyword.lower().strip()

        for anime in anime_table.values():
            # Prepare text fields for matching
            title = anime.title.lower() if anime.title else ""
            plot = anime.plot_synopsis.lower() if anime.plot_synopsis else ""

            # Calculate fuzzy matching scores
            # partial_ratio is useful for finding keywords within longer strings (like titles or plots)
            title_score = fuzz.partial_ratio(search_keyword, title)
            plot_score = fuzz.partial_ratio(search_keyword, plot)

            # Determine the best score for this entry
            best_score = max(title_score, plot_score)

            # Exact substring matches in the title should be prioritized
            if search_keyword in title:
                best_score = max(best_score, 90)

            # If the score meets a reasonable threshold, consider it a match
            if best_score >= 60:
                # Extract the year from the release_date datetime object
                release_year = None
                if anime.release_date:
                    release_year = anime.release_date.year

                scored_results.append({
                    "score": best_score,
                    "anime_id": anime.anime_id,
                    "title": anime.title,
                    "release_year": release_year
                })

        # Sort results by score in descending order (most relevant first)
        scored_results.sort(key=lambda x: x["score"], reverse=True)

        # Prepare the final list based on the requested limit
        final_results = []
        for item in scored_results[:limit]:
            final_results.append({
                "anime_id": item["anime_id"],
                "title": item["title"],
                "release_year": item["release_year"]
            })

        return {"results": final_results}

    @is_tool()
    def get_movie_cast_list(self, content_id: str):
        """
        Retrieve the complete cast list for a specific movie, including actor names, 
        character names, and billing orders.
        """
        # Access the database tables for cast members and persons
        # Using getattr to safely access the tables from the db instance
        cast_table = getattr(self.db, 'cast_member', {})
        person_table = getattr(self.db, 'person', {})

        # Ensure the tables are dictionaries and not None
        if cast_table is None:
            cast_table = {}
        if person_table is None:
            person_table = {}

        # Initialize the list to store formatted cast information
        cast_results = []
        found_any = False

        # Iterate through all cast member records to find matches for the given content_id
        # content_id is an identifier, so we use exact matching (==)
        for cast_member in cast_table.values():
            if cast_member.content_id == content_id:
                found_any = True

                # Retrieve the person's name using the person_id foreign key
                person_id = cast_member.person_id
                person_record = person_table.get(person_id)

                # Use the person's name if found, otherwise default to "Unknown"
                actor_name = person_record.name if person_record else "Unknown"

                # Construct the individual cast member entry based on the return schema
                # Ensure character_name is a string as per return schema requirements
                cast_entry = {
                    "actor_name": actor_name,
                    "character_name": cast_member.character_name if cast_member.character_name else "Unknown",
                    "billing_order": cast_member.billing_order
                }
                cast_results.append(cast_entry)

        # The tool schema specifies a pre-condition that the Movie ID must exist and raises KeyError.
        # If no records are found, we raise KeyError as per the tool schema requirement.
        if not found_any:
            raise KeyError(f"Movie with ID '{content_id}' not found or has no cast records.")

        # Sort the cast list by billing order to ensure the result matches standard movie credits.
        # We use a tuple (is_none, value) to ensure None values are sorted to the end 
        # and to avoid TypeError in Python 3 when comparing multiple None values during sorting.
        cast_results.sort(key=lambda x: (x['billing_order'] is None, x['billing_order'] if x['billing_order'] is not None else 0))

        # Return the result wrapped in the 'cast' property as defined in the return schema
        return {"cast": cast_results}

    @is_tool()
    def get_top_rated_movies(self, limit: int, min_vote_count: int = None):
        """
        Retrieve a list of top-rated movies sorted by average rating in descending order.

        Args:
            limit (int): Maximum number of top-rated movies to return.
            min_vote_count (int, optional): Minimum number of votes required for a movie to be included.

        Returns:
            dict: A dictionary containing the list of top-rated movies.

        Raises:
            ValueError: If parameters are invalid.
        """
        # Parameter validation
        if limit <= 0:
            raise ValueError("Limit must be a positive integer.")

        # Access the database
        db = self.db
        movie_table = getattr(db, "movie", {})

        if not movie_table:
            return {"top_rated_movies": []}

        # Convert dictionary values to a list of Movie objects
        movies = list(movie_table.values())

        # Filter movies based on the minimum vote count if provided
        filtered_movies = []
        for movie in movies:
            # Check if the movie has a rating and vote count
            rating = movie.average_rating if movie.average_rating is not None else 0.0
            votes = movie.vote_count if movie.vote_count is not None else 0

            # Apply min_vote_count filter
            if min_vote_count is not None:
                if votes < min_vote_count:
                    continue

            filtered_movies.append(movie)

        # Sort movies by average_rating in descending order
        # If ratings are equal, we can optionally sort by vote_count or title for stability
        sorted_movies = sorted(
            filtered_movies, 
            key=lambda x: (x.average_rating if x.average_rating is not None else 0.0, 
                           x.vote_count if x.vote_count is not None else 0), 
            reverse=True
        )

        # Apply the limit
        top_movies = sorted_movies[:limit]

        # Format the results according to the tool schema
        result_list = []
        for m in top_movies:
            result_list.append({
                "movie_id": m.movie_id,
                "title": m.title,
                "average_rating": float(m.average_rating) if m.average_rating is not None else 0.0,
                "vote_count": int(m.vote_count) if m.vote_count is not None else 0
            })

        return {"top_rated_movies": result_list}

    @is_tool()
    def get_related_anime(self, content_id: str, limit: int = 10):
        """
        Retrieve a list of anime related to a specified anime based on similarity in genre, studio, staff, or themes.

        Args:
            content_id (str): Unique identifier for the reference anime.
            limit (int): Maximum number of related anime to return. Defaults to 10.

        Returns:
            dict: A dictionary containing a list of related anime with their similarity scores.

        Raises:
            KeyError: If the content_id does not exist in the anime database.
        """
        # 1. Access the database tables
        # Note: Using 'or {}' because getattr with a default of {} on an Optional field that is None returns None.
        anime_dict = getattr(self.db, 'anime') or {}
        content_genre_dict = getattr(self.db, 'content_genre') or {}
        crew_member_dict = getattr(self.db, 'crew_member') or {}
        cast_member_dict = getattr(self.db, 'cast_member') or {}
        content_production_company_dict = getattr(self.db, 'content_production_company') or {}

        # 2. Check if the source anime exists (Pre-condition check)
        if content_id not in anime_dict:
            raise KeyError(f"Anime with ID '{content_id}' not found.")

        # 3. Pre-index features for all anime to optimize calculation from O(N*M) to O(N+M)
        # This prevents scanning the entire link tables repeatedly inside the comparison loop.
        features_map = {aid: {'genres': set(), 'studios': set(), 'people': set()} for aid in anime_dict}

        # Index Genres
        for cg in content_genre_dict.values():
            if cg.content_id in features_map:
                features_map[cg.content_id]['genres'].add(cg.genre_id)

        # Index Studios (from both Anime table property and ProductionCompany link table)
        for aid, anime in anime_dict.items():
            if anime.company_name:
                features_map[aid]['studios'].add(anime.company_name)
        for cpc in content_production_company_dict.values():
            if cpc.content_id in features_map:
                features_map[cpc.content_id]['studios'].add(cpc.company_id)

        # Index People (Staff and Cast)
        for cm in crew_member_dict.values():
            if cm.content_id in features_map:
                features_map[cm.content_id]['people'].add(cm.person_id)
        for cm in cast_member_dict.values():
            if cm.content_id in features_map:
                features_map[cm.content_id]['people'].add(cm.person_id)

        # 4. Get source features for the reference anime
        src = features_map[content_id]
        src_genres = src['genres']
        src_studios = src['studios']
        src_people = src['people']

        # 5. Calculate similarity for all other anime in the database
        results = []
        for target_id, tar in features_map.items():
            if target_id == content_id:
                continue

            # Genre Similarity (Jaccard Index)
            genre_score = 0.0
            union_genres = src_genres | tar['genres']
            if union_genres:
                genre_score = len(src_genres & tar['genres']) / len(union_genres)

            # Studio Similarity (Binary: share at least one studio or production company)
            studio_score = 1.0 if (src_studios & tar['studios']) else 0.0

            # Staff/Cast Similarity (Jaccard Index)
            people_score = 0.0
            union_people = src_people | tar['people']
            if union_people:
                people_score = len(src_people & tar['people']) / len(union_people)

            # Total Weighted Similarity Score
            # Weights: Genres (40%), Studio (30%), Staff/Cast (30%)
            total_score = (genre_score * 0.4) + (studio_score * 0.3) + (people_score * 0.3)

            # Only include anime with some degree of similarity
            if total_score > 0:
                results.append({
                    "anime_id": target_id,
                    "title": anime_dict[target_id].title,
                    "similarity_score": round(total_score, 3)
                })

        # 6. Sort by score descending and apply the requested limit
        results.sort(key=lambda x: x['similarity_score'], reverse=True)

        return {
            "related_anime": results[:limit]
        }

    @is_tool()
    def filter_movies_by_release_year_range(self, start_year: int, end_year: int, limit: int = None):
            """
            Filter and retrieve movies released within a specified year range.
            """
            # Pre-condition check: Ensure the start year is less than or equal to the end year
            if start_year > end_year:
                raise ValueError(f"Pre-condition failed: start_year ({start_year}) must be less than or equal to end_year ({end_year}).")

            # Access the movie table from the database
            # According to the database class definition, movies are stored in a dictionary keyed by movie_id
            movie_table = getattr(self.db, "movie", {})
            if not movie_table:
                return {"results": []}

            filtered_results = []

            # Iterate through all movie records in the database
            for movie in movie_table.values():
                # The database stores release_date as a datetime object or None
                if movie.release_date:
                    # Extract the year from the release_date datetime object
                    movie_year = movie.release_date.year

                    # Check if the movie's release year is within the inclusive range [start_year, end_year]
                    if start_year <= movie_year <= end_year:
                        # Construct the result object matching the tool schema's return item definition
                        movie_info = {
                            "movie_id": movie.movie_id,
                            "title": movie.title,
                            "release_year": movie_year
                        }
                        filtered_results.append(movie_info)

            # Sort results by release year (ascending) and then title (alphabetical) for consistent output
            filtered_results.sort(key=lambda x: (x["release_year"], x["title"]))

            # Apply the limit parameter if provided and valid
            if limit is not None and limit > 0:
                filtered_results = filtered_results[:limit]

            # Return the results wrapped in the expected dictionary structure
            return {"results": filtered_results}

    @is_tool()
    def get_movie_production_companies(self, content_id: str):
        # Access the database instance from the self object
        db = self.db

        # Safely retrieve the content-production mapping table and the company details table
        # Using getattr to handle cases where the table might not be initialized
        # We use 'or {}' to handle cases where the attribute exists but is None
        content_links = getattr(db, "content_production_company", {}) or {}
        companies_info = getattr(db, "production_company", {}) or {}

        # Initialize a list to hold the resulting production company details
        production_companies = []

        # Flag to verify the existence of the movie ID in our records, as per the pre-condition
        movie_exists_in_records = False

        # Iterate through the mapping table to find all companies associated with the given content_id
        # content_id is a unique identifier, so we use exact matching (==)
        for link in content_links.values():
            if link.content_id == content_id:
                movie_exists_in_records = True

                # Retrieve the detailed information for the specific company using its ID
                company_id = link.company_id
                company_details = companies_info.get(company_id)

                # If the company exists in the production_company table, extract the required fields
                if company_details:
                    production_companies.append({
                        "company_name": company_details.company_name,
                        "country": company_details.country
                    })

        # According to the tool schema's pre-condition and raises section:
        # If no record of the movie ID is found in the mapping table, we raise a KeyError.
        if not movie_exists_in_records:
            raise KeyError(f"The movie ID '{content_id}' was not found in the production database.")

        # Return the list of production companies in the format specified by the tool schema
        return {"production_companies": production_companies}

    @is_tool()
    def get_tv_series_cast_list(self, content_id: str):
        """
        Retrieve the complete cast list for a specific TV series, including actor names, 
        character names, and episode counts.
        """
        # Access the database tables for cast members and persons
        # Use 'or {}' to ensure we have a dictionary even if the attribute value is None
        cast_member_table = getattr(self.db, 'cast_member', {}) or {}
        person_table = getattr(self.db, 'person', {}) or {}

        # Filter the cast_member records to find those associated with the specific content_id
        # Since content_id is a unique identifier, we use exact matching (==)
        series_cast_records = [
            cast for cast in cast_member_table.values() 
            if cast.content_id == content_id
        ]

        # If no cast records are found for the given content_id, raise a KeyError
        # as the pre-condition requires the TV series ID to exist in the database.
        if not series_cast_records:
            raise KeyError(f"TV series with ID '{content_id}' not found in the database.")

        cast_list = []
        for cast in series_cast_records:
            # Retrieve the person details using the person_id from the cast record
            person = person_table.get(cast.person_id)

            # Extract fields while handling potential None values in the database
            # actor_name is derived from the person table
            actor_name = person.name if person else "Unknown Actor"
            # character_name and episode_count are directly in the cast_member record
            character_name = cast.character_name if cast.character_name else "Unknown"
            episode_count = cast.episode_count if cast.episode_count is not None else 0

            cast_list.append({
                "actor_name": actor_name,
                "character_name": character_name,
                "episode_count": episode_count
            })

        # Return the results structured according to the tool schema
        return {"cast": cast_list}

    @is_tool()
    def get_movie_trailers_list(self, content_id: str):
        """
        Retrieve a list of trailers and promotional videos available for a specific movie.

        Args:
            content_id (str): Unique identifier for the movie (e.g., "mov_12345").

        Returns:
            dict: A dictionary containing a list of trailers with their details.

        Raises:
            KeyError: If the movie content_id is not found in the database.
        """
        from datetime import datetime

        # Access the media_asset table from the database
        # The database is instantiated in self.db, and we access the media_asset attribute
        media_asset_table = getattr(self.db, "media_asset", None)

        if media_asset_table is None:
            # If the table is not found or not initialized, raise a KeyError
            raise KeyError("Media asset database table is not available.")

        trailers_list = []
        content_exists = False

        # Iterate through all media assets to find those matching the content_id
        # We use exact matching for content_id as it is a unique identifier
        for asset_id, asset in media_asset_table.items():
            if asset.content_id == content_id:
                content_exists = True

                # Filter specifically for assets of type "trailer"
                if asset.asset_type == "trailer":
                    # Convert the datetime object to the required "yyyy-mm-dd HH:MM:SS" string format
                    formatted_release_date = ""
                    if asset.release_date:
                        formatted_release_date = asset.release_date.strftime("%Y-%m-%d %H:%M:%S")

                    # Construct the trailer information dictionary based on tool schema returns
                    trailer_info = {
                        "trailer_title": asset.title if asset.title else "Untitled Trailer",
                        "video_url": asset.url,
                        "duration_seconds": asset.runtime if asset.runtime is not None else 0,
                        "release_date": formatted_release_date
                    }
                    trailers_list.append(trailer_info)

        # Pre-condition check: Movie ID must exist in the database
        # If no assets were found for the given content_id, raise a KeyError
        if not content_exists:
            raise KeyError(f"Movie content ID '{content_id}' does not exist in the database.")

        # Return the list of trailers wrapped in the specified dictionary format
        return {"trailers": trailers_list}

    @is_tool()
    def calculate_average_rating_for_movie_list(self, content_ids: list[str]):
        """
        Calculate the average rating across a given list of movie IDs by querying the movie database.
        """
        # Access the movie table from the database
        # According to the tool definition, the movie data is stored in the 'movie' attribute of the DB
        movie_db = getattr(self.db, 'movie', {})

        # Check if the database table exists
        if movie_db is None:
            movie_db = {}

        # Initialize variables for calculation
        total_rating = 0.0
        movie_count = 0

        # Check if the input list is empty
        if not content_ids:
            return {
                "average_rating": 0.0,
                "movie_count": 0
            }

        # Iterate through the provided movie IDs
        for movie_id in content_ids:
            # Pre-condition: All movie IDs must exist in the database.
            # If a movie_id is not found, raise a KeyError as specified in the schema.
            if movie_id not in movie_db:
                raise KeyError(f"Movie ID '{movie_id}' was not found in the database.")

            # Retrieve the movie object
            movie_item = movie_db[movie_id]

            # Get the average rating for the movie. 
            # Since average_rating is Optional[float] in the schema, we handle None as 0.0.
            rating = movie_item.average_rating if movie_item.average_rating is not None else 0.0

            total_rating += float(rating)
            movie_count += 1

        # Calculate the overall average rating
        # We use a simple arithmetic mean: sum of ratings divided by the number of movies
        if movie_count > 0:
            final_average = total_rating / movie_count
        else:
            final_average = 0.0

        # Return the results as defined in the returns schema
        return {
            "average_rating": float(final_average),
            "movie_count": int(movie_count)
        }

    @is_tool()
    def filter_tv_series_by_episode_count_range(self, min_episode_count: int, max_episode_count: int, limit: int = None):
            """
            Filter and retrieve TV series with total episode count within a specified range.

            Args:
                min_episode_count (int): Minimum number of episodes (inclusive).
                max_episode_count (int): Maximum number of episodes (inclusive).
                limit (int, optional): Maximum number of results to return.

            Returns:
                dict: A dictionary containing the list of filtered TV series.

            Raises:
                ValueError: If the input constraints are not met.
            """
            # 1. Validate pre-conditions: min <= max and both are positive integers
            if min_episode_count > max_episode_count:
                raise ValueError("Minimum episode count must be less than or equal to maximum episode count.")

            if min_episode_count <= 0 or max_episode_count <= 0:
                raise ValueError("Episode counts must be positive integers.")

            # 2. Access the tv_series data from the EntertainmentMediaQueryDB
            # The database provides a 'tv_series' attribute which is a dictionary of series_id: TvSeries objects
            tv_series_data = getattr(self.db, "tv_series", None)

            # If the database table is empty or doesn't exist, return an empty results list
            if not tv_series_data:
                return {"results": []}

            filtered_series = []

            # 3. Iterate through the TV series and filter by episode_count
            # We use .values() to iterate over the TvSeries model instances
            for series in tv_series_data.values():
                # TvSeries schema defines 'episode_count' as the total number of episodes
                current_count = series.episode_count

                # Check if the episode_count is within the specified range [min, max]
                # We handle the Optional[int] case by checking if current_count is not None
                if current_count is not None and min_episode_count <= current_count <= max_episode_count:
                    # Construct the result object matching the tool schema return fields
                    filtered_series.append({
                        "series_id": series.series_id,
                        "title": series.title,
                        "number_of_episodes": current_count
                    })

            # 4. Apply the result limit if specified
            if limit is not None and limit > 0:
                filtered_series = filtered_series[:limit]

            # 5. Return the results in the required format
            return {"results": filtered_series}

    @is_tool()
    def get_recently_added_anime(self, limit: int, days_back: int = None):
        """
        Retrieve a list of anime that were recently added to the database, sorted by addition date in descending order.
        """
        from datetime import datetime, timedelta

        # Input validation for limit
        if limit <= 0:
            raise ValueError("The limit must be a positive integer.")

        # Access the anime table from the database
        anime_table = getattr(self.db, "anime", None)
        if not anime_table:
            return {"recently_added": []}

        # Convert dictionary values to a list for processing
        anime_list = list(anime_table.values())

        # Filter by days_back if the parameter is provided
        filtered_anime = []
        if days_back is not None:
            if days_back < 0:
                raise ValueError("days_back must be a non-negative integer.")

            # Calculate the cutoff date based on the current time
            now = datetime.now()
            cutoff_date = now - timedelta(days=days_back)

            for anime in anime_list:
                # db.added_date is already a datetime object based on database_class definition
                if anime.added_date >= cutoff_date:
                    filtered_anime.append(anime)
        else:
            # If no days_back is specified, consider all anime for sorting
            filtered_anime = anime_list

        # Sort the anime list by added_date in descending order (most recent first)
        # We use a lambda to access the added_date attribute
        sorted_anime = sorted(
            filtered_anime, 
            key=lambda x: x.added_date, 
            reverse=True
        )

        # Apply the limit to the result set
        limited_anime = sorted_anime[:limit]

        # Format the results according to the return schema
        # Dates must be formatted as "yyyy-mm-dd HH:MM:SS"
        results = []
        for anime in limited_anime:
            results.append({
                "anime_id": anime.anime_id,
                "title": anime.title,
                "added_date": anime.added_date.strftime("%Y-%m-%d %H:%M:%S")
            })

        return {"recently_added": results}

    @is_tool()
    def get_anime_voice_cast_list(self, content_id: str):
            # Access the database from the object instance
            db = self.db

            # Retrieve the cast_member and person tables using getattr
            # cast_member: Dict[str, CastMember]
            # person: Dict[str, Person]
            cast_member_table = getattr(db, 'cast_member', None)
            person_table = getattr(db, 'person', None)

            # Ensure the required tables are available in the database
            if cast_member_table is None or person_table is None:
                raise KeyError("Required database tables 'cast_member' or 'person' are missing.")

            voice_cast_list = []
            found_anime = False

            # Iterate through the cast_member dictionary values to find matches for the content_id
            for member in cast_member_table.values():
                # Use exact matching for the content_id as it is a unique identifier
                if member.content_id == content_id:
                    found_anime = True

                    # The tool specifically requests the voice cast list
                    # We filter entries where the role is 'voice_actor'
                    if member.role == "voice_actor":
                        # Retrieve the person's name by looking up the person_id in the person table
                        person_record = person_table.get(member.person_id)
                        actor_name = person_record.name if person_record else "Unknown Actor"

                        # Build the cast member object with the required fields
                        cast_member_info = {
                            "voice_actor_name": actor_name,
                            "character_name": member.character_name if member.character_name else "N/A",
                            "language": member.language if member.language else "Unknown"
                        }
                        voice_cast_list.append(cast_member_info)

            # Per the pre-condition and schema requirements, if the content_id is not found, raise a KeyError
            if not found_anime:
                raise KeyError(f"Anime with content_id '{content_id}' not found in the database.")

            # Return the results wrapped in the voice_cast key as defined in the tool schema
            return {"voice_cast": voice_cast_list}

    @is_tool()
    def get_movie_details(self, content_id: str):
        """
        Retrieve complete metadata for a specific movie including title, plot synopsis, 
        cast, crew, release date, runtime, genres, and ratings.
        """
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Retrieve the movies table from the database
        # The database stores tables as dictionaries keyed by their instance keys (movie_id for Movie)
        movies_table = getattr(db, 'movie', None)

        # Pre-condition check: Movie ID must exist in the database
        if movies_table is None or content_id not in movies_table:
            raise KeyError(f"Movie with ID '{content_id}' not found in the database.")

        # Get the movie object
        movie = movies_table[content_id]

        # Initialize containers for related data
        genres_list = []
        cast_list = []
        crew_list = []

        # Fetch genres associated with this content_id from the content_genre table
        content_genres = getattr(db, 'content_genre', None)
        if content_genres:
            for genre_link in content_genres.values():
                if genre_link.content_id == content_id:
                    # genre_id represents the genre identifier/name in this schema
                    genres_list.append(genre_link.genre_id)

        # Fetch cast members associated with this content_id
        cast_members_table = getattr(db, 'cast_member', None)
        if cast_members_table:
            for cast_member in cast_members_table.values():
                if cast_member.content_id == content_id:
                    cast_list.append({
                        "actor_name": cast_member.person_id,
                        "character_name": cast_member.character_name
                    })

        # Fetch crew members associated with this content_id
        crew_members_table = getattr(db, 'crew_member', None)
        if crew_members_table:
            for crew_member in crew_members_table.values():
                if crew_member.content_id == content_id:
                    crew_list.append({
                        "crew_name": crew_member.person_id,
                        "role": crew_member.role
                    })

        # Format the release date to 'yyyy-mm-dd' string format as required by the schema
        release_date_str = None
        if movie.release_date:
            # Based on background info, db time objects are datetime.datetime
            release_date_str = movie.release_date.strftime("%Y-%m-%d")

        # Construct and return the complete metadata dictionary
        return {
            "movie_id": movie.movie_id,
            "title": movie.title,
            "plot_synopsis": movie.plot_synopsis,
            "release_date": release_date_str,
            "runtime_minutes": movie.runtime,
            "genres": genres_list,
            "average_rating": movie.average_rating,
            "cast": cast_list,
            "crew": crew_list
        }

    @is_tool()
    def filter_tv_series_by_air_year_range(self, start_year: int, end_year: int, limit: int = None):
        """
        Filter and retrieve TV series that first aired within a specified year range.

        Args:
            start_year (int): Starting year of the range (inclusive).
            end_year (int): Ending year of the range (inclusive).
            limit (int, optional): Maximum number of results to return.

        Returns:
            dict: A dictionary containing a list of TV series results.

        Raises:
            ValueError: If start_year is greater than end_year.
        """
        # Pre-condition check: Start year must be less than or equal to end year
        if start_year > end_year:
            raise ValueError(f"Invalid year range: start_year ({start_year}) must be less than or equal to end_year ({end_year}).")

        # Access the database
        db = self.db
        tv_series_table = getattr(db, 'tv_series', None)

        results = []

        # If the table doesn't exist or is empty, return an empty results list
        if not tv_series_table:
            return {"results": []}

        # Iterate through all TV series records
        for series_id, series_obj in tv_series_table.items():
            # Check if release_date (first air date) is available
            if series_obj.release_date:
                # Extract the year from the datetime object
                first_air_year = series_obj.release_date.year

                # Check if the year falls within the inclusive range
                if start_year <= first_air_year <= end_year:
                    results.append({
                        "series_id": series_obj.series_id,
                        "title": series_obj.title,
                        "first_air_year": first_air_year
                    })

        # Sort results by year for better presentation (optional but helpful)
        results.sort(key=lambda x: x['first_air_year'])

        # Apply the limit if provided
        if limit is not None and limit > 0:
            results = results[:limit]

        return {"results": results}

    @is_tool()
    def search_anime_by_studio(self, studio_name: str, limit: int = None):
        """
        Search for anime produced by a specific animation studio by studio name.

        Args:
            studio_name: Full or partial name of the animation studio to search for.
            limit: Maximum number of results to return.

        Returns:
            A dictionary containing a list of anime produced by the specified studio.
        """
        # Pre-condition check: Studio name must be a non-empty string
        if not studio_name or not isinstance(studio_name, str) or not studio_name.strip():
            raise ValueError("Studio name must be a non-empty string")

        # Access the anime database table
        db = self.db
        anime_table = getattr(db, "anime", {})
        if not anime_table:
            return {"anime": []}

        matches = []

        # Iterate through all anime records to find matches based on the animation studio (company_name)
        for anime_id, anime_obj in anime_table.items():
            # Skip if the anime has no associated company_name
            if not anime_obj.company_name:
                continue

            # Use fuzzy matching for the studio name as it is a natural language text field.
            # fuzz.partial_ratio is used to handle partial matches (e.g., "Pierrot" for "Studio Pierrot").
            score = fuzz.partial_ratio(studio_name.lower(), anime_obj.company_name.lower())

            # Use a threshold to filter relevant results. 
            # A score of 85 is generally robust for partial matches in names.
            if score >= 85:
                matches.append((anime_obj, score))

        # Sort the matches by similarity score in descending order to return the most relevant results first
        matches.sort(key=lambda x: x[1], reverse=True)

        # Prepare the response list
        anime_list = []
        for anime_obj, score in matches:
            # Extract the year from the release_date datetime object
            first_air_year = None
            if anime_obj.release_date:
                first_air_year = anime_obj.release_date.year

            anime_list.append({
                "anime_id": anime_obj.anime_id,
                "title": anime_obj.title,
                "first_air_year": first_air_year
            })

            # Apply the limit if specified
            if limit is not None and len(anime_list) >= limit:
                break

        return {"anime": anime_list}

    @is_tool()
    def get_movie_release_schedule(self, start_date: str, end_date: str, limit: int = None):
            from datetime import datetime

            # Validate the date format and logic
            try:
                # Parse the input strings into datetime objects for comparison
                # According to the instructions, "yyyy-mm-dd" represents the date only
                start_dt = datetime.strptime(start_date, "%Y-%m-%d")
                end_dt = datetime.strptime(end_date, "%Y-%m-%d")
            except ValueError:
                raise ValueError("Dates must be in 'yyyy-mm-dd' format.")

            # Pre-condition check: Start date must be before or equal to end date
            if start_dt > end_dt:
                raise ValueError("Start date must be before or equal to end date.")

            # Access the movie table from the database
            movie_table = getattr(self.db, "movie", {})
            if not movie_table:
                return {"scheduled_movies": []}

            scheduled_movies = []

            # Iterate through all movies in the database
            for movie_id, movie in movie_table.items():
                # Check if the movie has a release date assigned
                if movie.release_date:
                    # Compare only the date component to ensure the entire day is covered
                    movie_date = movie.release_date.date()
                    if start_dt.date() <= movie_date <= end_dt.date():
                        scheduled_movies.append({
                            "movie_id": movie.movie_id,
                            "title": movie.title,
                            # Format the return date as yyyy-mm-dd as per the schema examples
                            "release_date": movie.release_date.strftime("%Y-%m-%d")
                        })

            # Sort movies by release date for a chronological schedule
            scheduled_movies.sort(key=lambda x: x["release_date"])

            # Apply the limit if provided and valid
            if limit is not None and limit > 0:
                scheduled_movies = scheduled_movies[:limit]

            # Post-condition: Returns list of movies with release dates in the specified range
            return {"scheduled_movies": scheduled_movies}

    @is_tool()
    def filter_anime_by_episode_count_range(self, min_episode_count: int, max_episode_count: int, limit: int = None):
        """
        Filters and retrieves anime with total episode count within a specified range.

        Args:
            min_episode_count (int): Minimum number of episodes (inclusive).
            max_episode_count (int): Maximum number of episodes (inclusive).
            limit (int, optional): Maximum number of results to return.

        Returns:
            dict: A dictionary containing a list of anime matching the criteria.
        """
        # 1. Validate pre-conditions: episode counts must be positive integers (> 0) and min <= max
        if min_episode_count <= 0 or max_episode_count <= 0:
            raise ValueError("Episode counts must be positive integers (greater than zero).")

        if min_episode_count > max_episode_count:
            raise ValueError("Minimum episode count must be less than or equal to maximum episode count.")

        # 2. Access the database instance and the anime table
        db = self.db
        # Using getattr to safely handle cases where the table might not be initialized or is None
        anime_table = getattr(db, "anime", None)

        if not anime_table:
            # If the table doesn't exist or is empty, return an empty result list
            return {"results": []}

        # 3. Filter anime based on the episode count range
        filtered_results = []
        for anime in anime_table.values():
            # Check if the episode_count field is present and within the specified range (inclusive)
            if anime.episode_count is not None:
                if min_episode_count <= anime.episode_count <= max_episode_count:
                    # Map database fields to the tool's return structure defined in the schema
                    filtered_results.append({
                        "anime_id": anime.anime_id,
                        "title": anime.title,
                        "number_of_episodes": anime.episode_count
                    })

        # 4. Apply the result limit if specified
        if limit is not None:
            if limit < 0:
                raise ValueError("Limit must be a non-negative integer.")
            filtered_results = filtered_results[:limit]

        # 5. Return the formatted results according to the tool schema
        return {"results": filtered_results}

    @is_tool()
    def get_tv_series_production_companies(self, content_id: str):
        # Access the database instance from the class
        db = self.db

        # Retrieve the necessary tables from the database
        # content_production_company: maps content (TV series) to production companies
        # production_company: contains details like name and country for each company
        content_mapping = getattr(db, 'content_production_company', None) or {}
        companies_data = getattr(db, 'production_company', None) or {}

        # Initialize a flag to track if the series ID exists in our mapping database
        # and a list to store the results
        series_exists = False
        production_companies_result = []

        # Iterate through all records in the mapping table
        # Mapping records follow the ContentProductionCompany schema
        for record_id, mapping in content_mapping.items():
            if mapping.content_id == content_id:
                series_exists = True

                # Use the company_id from the mapping to look up the company details
                company_id = mapping.company_id
                company_info = companies_data.get(company_id)

                # If company details are found, append the required fields to our result list
                if company_info:
                    production_companies_result.append({
                        "company_name": company_info.company_name,
                        "country": company_info.country
                    })

        # The tool schema specifies a pre-condition that the TV series ID must exist.
        # If no mapping record was found for the provided content_id, raise a KeyError.
        if not series_exists:
            raise KeyError(f"TV series with ID '{content_id}' not found in the production database.")

        # Return the results in the format specified by the tool schema
        return {"production_companies": production_companies_result}

    @is_tool()
    def compare_movie_ratings(self, content_id_1: str, content_id_2: str):
        """
        Compare the ratings of two movies and return the comparison result.

        This method retrieves movie information from the database using the provided IDs.
        It verifies that both movies exist before performing the comparison as per the 
        tool's pre-condition.
        """
        # Access the database instance from the tool class
        db = self.db

        # Retrieve the 'movie' table from the database
        # Based on the database definition, this is a dictionary mapping movie_id to Movie objects
        movie_table = getattr(db, "movie", {})
        if movie_table is None:
            movie_table = {}

        # Retrieve the first movie record using its unique identifier
        # Since identifiers (IDs) are exact, fuzzy matching is not applied here
        movie_1 = movie_table.get(content_id_1)
        if not movie_1:
            # Pre-condition: Both movie IDs must exist in the database
            raise ValueError(f"Movie with ID '{content_id_1}' not found in the database.")

        # Retrieve the second movie record using its unique identifier
        movie_2 = movie_table.get(content_id_2)
        if not movie_2:
            # Pre-condition: Both movie IDs must exist in the database
            raise ValueError(f"Movie with ID '{content_id_2}' not found in the database.")

        # The tool schema specifies returning movie_1_title, movie_1_rating, and movie_2_title.
        # We extract these values from the Movie objects.
        # For ratings, we ensure the value is a number (float) or 0.0 if not set.
        movie_1_title = movie_1.title
        movie_1_rating = float(movie_1.average_rating) if movie_1.average_rating is not None else 0.0
        movie_2_title = movie_2.title

        # Construct the result dictionary matching the tool schema's return properties
        result = {
            "movie_1_title": movie_1_title,
            "movie_1_rating": movie_1_rating,
            "movie_2_title": movie_2_title
        }

        return result

    @is_tool()
    def get_tv_series_trailers_list(self, content_id: str):
        # Access the media_asset table from the database
        # The database instance is available at self.db
        media_assets = getattr(self.db, "media_asset", {})

        if media_assets is None:
            media_assets = {}

        trailers_list = []
        content_id_found = False

        # Iterate through all media assets to find trailers associated with the content_id
        for asset in media_assets.values():
            if asset.content_id == content_id:
                # Mark that the content_id exists in the database to satisfy pre-condition check
                content_id_found = True

                # Check if the asset is a trailer
                if asset.asset_type == "trailer":
                    # Handle release_date formatting
                    # All date-time parameters and return values must use "yyyy-mm-dd HH:MM:SS" format
                    formatted_release_date = ""
                    if asset.release_date:
                        formatted_release_date = asset.release_date.strftime("%Y-%m-%d %H:%M:%S")

                    # Construct the trailer object according to the tool schema
                    trailer_info = {
                        "trailer_title": asset.title if asset.title else "Untitled Trailer",
                        "video_url": asset.url,
                        "duration_seconds": asset.runtime if asset.runtime is not None else 0,
                        "release_date": formatted_release_date
                    }
                    trailers_list.append(trailer_info)

        # Pre-condition: TV series ID must exist in the database.
        # If no assets (of any type) were found for the content_id, raise KeyError.
        if not content_id_found:
            raise KeyError(f"TV series ID '{content_id}' not found in the database.")

        # Return the list of trailers wrapped in the expected return structure
        return {"trailers": trailers_list}

    @is_tool()
    def get_top_rated_tv_series(self, limit: int, min_vote_count: int = None):
        """
        Retrieve a list of top-rated TV series sorted by average rating in descending order.
        """
        # Validation for input parameters as per 'raises' requirement in schema
        if not isinstance(limit, int) or limit <= 0:
            raise ValueError("limit must be a positive integer.")

        if min_vote_count is not None:
            if not isinstance(min_vote_count, int) or min_vote_count < 0:
                raise ValueError("min_vote_count must be a non-negative integer.")

        # Access the tv_series table from the database
        # getattr(self.db, "tv_series") returns the dict or None
        tv_series_table = getattr(self.db, "tv_series", None)

        if not tv_series_table:
            return {"top_rated_series": []}

        filtered_series = []

        # Iterate through all TV series in the database
        for series in tv_series_table.values():
            # Check if the series has a rating; if not, it shouldn't be included
            if series.average_rating is None:
                continue

            # Apply the minimum vote count filter if provided
            if min_vote_count is not None:
                # If vote_count is missing, we assume it doesn't meet the threshold
                if series.vote_count is None or series.vote_count < min_vote_count:
                    continue

            filtered_series.append(series)

        # Sort the series by average_rating in descending order
        # In case of ties in rating, we sort by vote_count as a secondary criterion
        filtered_series.sort(
            key=lambda x: (x.average_rating, x.vote_count if x.vote_count is not None else 0), 
            reverse=True
        )

        # Apply the limit to get the top N series
        top_rated_results = filtered_series[:limit]

        # Format the output according to the return schema
        formatted_series_list = []
        for s in top_rated_results:
            formatted_series_list.append({
                "series_id": s.series_id,
                "title": s.title,
                "average_rating": s.average_rating,
                "vote_count": s.vote_count if s.vote_count is not None else 0
            })

        return {"top_rated_series": formatted_series_list}

    @is_tool()
    def get_movie_crew_list(self, content_id: str):
        """
        Retrieve the complete crew list for a specific movie, including crew member names, roles, and departments.

        Args:
            content_id (str): Unique identifier for the movie.

        Returns:
            dict: A dictionary containing the list of crew members.

        Raises:
            KeyError: If the movie with the given content_id is not found in the records.
        """
        # Access the database instance
        db = self.db

        # Retrieve the crew_member and person tables from the database
        # Using getattr with a default of {} to handle cases where the table might not be initialized or is None
        crew_member_data = getattr(db, "crew_member", {}) or {}
        person_data = getattr(db, "person", {}) or {}

        crew_results = []
        movie_found = False

        # Iterate through all entries in the crew_member table
        # content_id is an identifier (ID), so we use exact matching as per instructions
        for member in crew_member_data.values():
            if member.content_id == content_id:
                movie_found = True
                # For each matching crew entry, look up the person's name in the person table
                person_id = member.person_id
                person_info = person_data.get(person_id)

                # If the person exists, use their name; otherwise, default to 'Unknown'
                crew_name = person_info.name if person_info else "Unknown"

                # Construct the crew member object according to the tool schema
                # department is optional in DB but required as a string in return schema, so we default to ""
                crew_results.append({
                    "crew_name": crew_name,
                    "role": member.role,
                    "department": member.department if member.department else ""
                })

        # The pre-condition states the movie must exist and the schema specifies a KeyError should be raised
        if not movie_found:
            raise KeyError(f"Movie ID '{content_id}' not found in the database records.")

        # The tool schema specifies returning an object with a 'crew' property containing the array
        return {"crew": crew_results}

    @is_tool()
    def filter_tv_series_by_status(self, status: Literal["returning_series", "ended", "in_production", "canceled", "planned"], limit: int = None) -> Dict[str, List[Dict[str, str]]]:
        """
        Filter and retrieve TV series by their current production status (e.g., returning series, ended, in production, canceled).

        Args:
            status (Literal["returning_series", "ended", "in_production", "canceled", "planned"]): Current production status of the TV series.
            limit (int, optional): Maximum number of results to return.

        Returns:
            Dict[str, List[Dict[str, str]]]: A dictionary containing a list of matching TV series, each with its series_id, title, and status.

        Raises:
            ValueError: If the status provided is not valid according to the enum.
        """
        # Validate the status parameter against the allowed enum values
        valid_statuses = ["returning_series", "ended", "in_production", "canceled", "planned"]
        if status not in valid_statuses:
            raise ValueError(f"Invalid status: {status}. Must be one of {valid_statuses}")

        # Access the database and the tv_series table
        db = self.db
        tv_series_table = getattr(db, "tv_series", {})
        if not tv_series_table:
            return {"results": []}

        results = []
        # Iterate through all TV series in the database
        for series_id, series_obj in tv_series_table.items():
            # Check if the series status matches the requested status
            # Note: series_obj.status is an attribute of the TvSeries class
            if series_obj.status == status:
                results.append({
                    "series_id": series_obj.series_id,
                    "title": series_obj.title,
                    "status": series_obj.status
                })

        # Apply the limit if provided and valid
        if limit is not None and limit > 0:
            results = results[:limit]

        return {"results": results}

    @is_tool()
    def get_movie_awards_list(self, content_id: str):
            # Access the award table from the database instance.
            # getattr is used to safely access the 'award' table, defaulting to an empty dict if not found.
            award_table = getattr(self.db, "award", {})

            # If the award table is None (as it is Optional in the database schema), initialize it as an empty dict.
            if award_table is None:
                award_table = {}

            # Initialize a list to hold the award data for the specified movie.
            movie_awards_list = []

            # Iterate through all award entries in the database table.
            # Each entry in award_table is an instance of the Award class.
            for award_id, award_entry in award_table.items():
                # Check for a match on the content_id.
                # We use exact matching (==) for identifiers like content_id as per the instructions.
                if award_entry.content_id == content_id:
                    # Construct the award object according to the tool schema's return properties.
                    # result is an enum value ("Won" or "Nominated") defined in the Award schema.
                    award_data = {
                        "award_name": award_entry.award_name,
                        "category": award_entry.category,
                        "result": award_entry.result,
                        "year": award_entry.year
                    }
                    movie_awards_list.append(award_data)

            # Return the result dictionary containing the list of awards and nominations.
            # If no awards are found for the given content_id, an empty list is returned.
            return {"awards": movie_awards_list}

    @is_tool()
    def filter_anime_by_air_year_range(self, start_year: int, end_year: int, limit: int = None):
        """
        Filter and retrieve anime that first aired within a specified year range.

        Args:
            start_year (int): Starting year of the range (inclusive).
            end_year (int): Ending year of the range (inclusive).
            limit (int, optional): Maximum number of results to return.

        Returns:
            dict: A dictionary containing a list of anime results matching the criteria.

        Raises:
            ValueError: If start_year is greater than end_year.
        """
        # 1. Pre-condition check: Start year must be less than or equal to end year
        if start_year > end_year:
            raise ValueError(f"Start year ({start_year}) must be less than or equal to end year ({end_year}).")

        # 2. Access the anime database
        db = self.db
        anime_table = getattr(db, "anime", {})
        if not anime_table:
            return {"results": []}

        # 3. Filter anime records by year range
        # It is more efficient to filter before sorting and limiting
        filtered_anime = []
        for anime_obj in anime_table.values():
            if anime_obj.release_date:
                # Extract the year from the datetime object
                first_air_year = anime_obj.release_date.year

                # Check if the year falls within the inclusive range
                if start_year <= first_air_year <= end_year:
                    filtered_anime.append(anime_obj)

        # 4. Sort the filtered results by release_date for consistent output
        filtered_anime.sort(key=lambda x: x.release_date if x.release_date else datetime.min)

        # 5. Apply the limit and format the results
        # Using list slicing handles limit=None correctly ([:None] returns the whole list)
        # and also correctly handles limit=0 (returns an empty list).
        results = []
        for anime_obj in filtered_anime[:limit]:
            results.append({
                "anime_id": anime_obj.anime_id,
                "title": anime_obj.title,
                "first_air_year": anime_obj.release_date.year
            })

        return {"results": results}

    @is_tool()
    def filter_anime_by_broadcast_status(self, status: Literal["currently_airing", "finished_airing", "not_yet_aired"], limit: int = None):
            """
            Filters and retrieves anime from the database based on their current broadcast status.

            Args:
                status: The broadcast status to filter by (currently_airing, finished_airing, or not_yet_aired).
                limit: Maximum number of results to return.

            Returns:
                A dictionary containing a list of matching anime records.
            """
            # Validate that the status is one of the allowed enum values
            valid_statuses = ["currently_airing", "finished_airing", "not_yet_aired"]
            if status not in valid_statuses:
                raise ValueError(f"Invalid status: '{status}'. Expected one of {valid_statuses}.")

            # Access the entertainment media database
            db = self.db

            # Retrieve the anime table. getattr is used to safely access the table, 
            # defaulting to an empty dict if the table doesn't exist or is None.
            anime_table = getattr(db, "anime", {})
            if anime_table is None:
                anime_table = {}

            results = []

            # Iterate through the anime collection to find matches
            # anime_table is expected to be a Dict[str, Anime]
            for anime_id, anime_entry in anime_table.items():
                # Compare the anime's status with the requested status.
                # Since 'status' is a categorical/enum field, we use exact matching.
                if anime_entry.status == status:
                    results.append({
                        "anime_id": anime_entry.anime_id,
                        "title": anime_entry.title,
                        "broadcast_status": anime_entry.status
                    })

            # Apply the result limit if specified
            if limit is not None:
                if not isinstance(limit, int) or limit < 0:
                    raise ValueError("The 'limit' parameter must be a non-negative integer.")
                results = results[:limit]

            # Return the results in the format specified by the tool schema
            return {"results": results}

    @is_tool()
    def get_movie_alternative_titles(self, content_id: str):
        # Access the alternative_title table from the database
        # self.db is the EntertainmentMediaQueryDB instance provided in the context
        alt_titles_db = getattr(self.db, 'alternative_title', {})

        # If the database table is None, handle it as an empty dictionary to avoid iteration errors
        if alt_titles_db is None:
            alt_titles_db = {}

        # Initialize a list to store the matching alternative title information
        alternative_titles_list = []

        # Iterate through all records in the alternative_title table
        # Each record is an instance of the AlternativeTitle class
        for title_record in alt_titles_db.values():
            # Perform an exact match on content_id as it is a unique identifier (not a natural language field)
            if title_record.content_id == content_id:
                # Append the required fields to the results list
                alternative_titles_list.append({
                    "title": title_record.title,
                    "language": title_record.language,
                    "region": title_record.region
                })

        # The tool schema specifies a pre-condition that the Movie ID must exist in the database.
        # Additionally, the 'raises' field indicates that a KeyError should be raised.
        # If no records match the content_id, we raise a KeyError.
        if not alternative_titles_list:
            raise KeyError(f"Movie ID '{content_id}' not found in the database or has no alternative titles.")

        # Return the results wrapped in the structure defined by the tool schema's returns property
        return {
            "alternative_titles": alternative_titles_list
        }

    @is_tool()
    def get_anime_trailers_list(self, content_id: str):
        """
        Retrieve a list of trailers and promotional videos available for a specific anime from the media_asset database.
        """
        # Access the database instance from the class
        db = self.db

        # Retrieve the media_asset table from the database
        # media_asset is expected to be a dictionary where keys are asset IDs and values are MediaAsset objects
        media_assets = getattr(db, 'media_asset', {})
        if media_assets is None:
            media_assets = {}

        trailers_list = []
        # According to the pre-condition, the Anime ID must exist.
        # We use this flag to check if the content_id exists in any media asset record.
        anime_id_found = False

        # Iterate through all media assets to filter by content_id and asset_type
        for asset in media_assets.values():
            if asset.content_id == content_id:
                anime_id_found = True

                # Check if the asset is a trailer
                if asset.asset_type == "trailer":
                    # Handle the date formatting as required: "yyyy-mm-dd HH:MM:SS"
                    formatted_date = ""
                    if asset.release_date:
                        # asset.release_date is a datetime object in the database
                        formatted_date = asset.release_date.strftime("%Y-%m-%d %H:%M:%S")

                    # Construct the trailer object based on the tool schema's return requirements
                    trailer_info = {
                        "trailer_title": asset.title if asset.title else "Untitled Trailer",
                        "video_url": asset.url,
                        "duration_seconds": asset.runtime if asset.runtime is not None else 0,
                        "release_date": formatted_date
                    }
                    trailers_list.append(trailer_info)

        # If no records were found for the given content_id, raise a KeyError as specified in the schema
        if not anime_id_found:
            raise KeyError(f"Anime with ID '{content_id}' does not exist in the database.")

        # Return the results in the format specified by the tool schema
        return {"trailers": trailers_list}

    @is_tool()
    def get_tv_series_air_schedule(self, start_date: str, end_date: str, limit: int = 100):
        """
        Retrieve TV series scheduled to air within a specified date range.

        Args:
            start_date (str): Start date of the range in yyyy-mm-dd format.
            end_date (str): End date of the range in yyyy-mm-dd format.
            limit (int): Maximum number of results to return.

        Returns:
            dict: A dictionary containing the list of scheduled TV series.
        """
        from datetime import datetime

        # Helper function to parse dates robustly and handle end-of-day for range inclusion
        def parse_dt(date_str: str, is_end: bool = False) -> datetime:
            for fmt in ("%Y-%m-%d %H:%M:%S", "%Y-%m-%d"):
                try:
                    dt = datetime.strptime(date_str, fmt)
                    # If only a date is provided for the end of the range, include the whole day
                    if is_end and fmt == "%Y-%m-%d":
                        dt = dt.replace(hour=23, minute=59, second=59)
                    return dt
                except ValueError:
                    continue
            raise ValueError("Dates must be in yyyy-mm-dd or yyyy-mm-dd HH:MM:SS format")

        # 1. Parse the input date strings into datetime objects
        start_dt = parse_dt(start_date)
        end_dt = parse_dt(end_date, is_end=True)

        # 2. Pre-condition check: start_date must be before or equal to end_date
        if start_dt > end_dt:
            raise ValueError("The start_date must be before or equal to the end_date.")

        # 3. Access the database tables (only those defined in related_databases)
        db = self.db
        episodes = getattr(db, 'episode', {}) or {}
        tv_series_map = getattr(db, 'tv_series', {}) or {}

        # 4. Iterate through episodes to find those airing within the range
        scheduled_items = []

        for episode in episodes.values():
            if episode.release_date:
                # Check if the episode's release date falls within the specified range
                if start_dt <= episode.release_date <= end_dt:
                    series_id = episode.content_id
                    series_info = tv_series_map.get(series_id)

                    if series_info:
                        # Format the air_date according to the requirements:
                        # "yyyy-mm-dd HH:MM:SS" for full timestamp, "yyyy-mm-dd" for date only.
                        if (episode.release_date.hour == 0 and 
                            episode.release_date.minute == 0 and 
                            episode.release_date.second == 0):
                            air_date_str = episode.release_date.strftime("%Y-%m-%d")
                        else:
                            air_date_str = episode.release_date.strftime("%Y-%m-%d %H:%M:%S")

                        scheduled_items.append({
                            "air_date_obj": episode.release_date, # Used for sorting
                            "data": {
                                "series_id": series_info.series_id,
                                "title": series_info.title,
                                "air_date": air_date_str
                            }
                        })

        # 5. Sort the results chronologically by actual datetime object for accuracy
        scheduled_items.sort(key=lambda x: x["air_date_obj"])

        # 6. Apply the limit and extract the formatted data
        final_results = [item["data"] for item in scheduled_items[:limit]]

        return {"scheduled_series": final_results}

    @is_tool()
    def filter_movies_by_rating_range(self, min_rating: float, max_rating: float, limit: int = None):
        # 1. Validate the input parameters according to the tool's pre-conditions
        # Ratings must be within the valid range of 0 to 10
        if not (0 <= min_rating <= 10) or not (0 <= max_rating <= 10):
            raise ValueError("Invalid rating range: Ratings must be between 0 and 10.")

        # Minimum rating must be less than or equal to maximum rating
        if min_rating > max_rating:
            raise ValueError("Pre-condition failed: min_rating must be less than or equal to max_rating.")

        # 2. Access the movie table from the entertainment_media_query database
        # The database instance is accessible via self.db
        movie_table = getattr(self.db, 'movie', {})
        if not movie_table:
            # Return an empty result list if the movie table is missing or empty
            return {"results": []}

        # 3. Iterate through all movies and filter by the rating criteria
        filtered_movies = []
        for movie in movie_table.values():
            # Check if the movie has an average rating value
            if movie.average_rating is not None:
                # Check if the rating falls within the inclusive [min_rating, max_rating] range
                if min_rating <= movie.average_rating <= max_rating:
                    # Prepare the movie object according to the tool's return schema
                    movie_info = {
                        "movie_id": movie.movie_id,
                        "title": movie.title,
                        "average_rating": movie.average_rating
                    }
                    filtered_movies.append(movie_info)

        # 4. Optional: Sorting (usually helpful for users)
        # Although not explicitly required, we sort by rating descending for better relevance
        filtered_movies.sort(key=lambda x: x['average_rating'], reverse=True)

        # 5. Apply the result limit if provided
        # If limit is specified and positive, truncate the list
        if limit is not None and limit > 0:
            filtered_movies = filtered_movies[:limit]

        # 6. Return the results dictionary as defined in the tool schema
        return {"results": filtered_movies}

    @is_tool()
    def get_top_rated_anime(self, limit: int, min_vote_count: int = None):
        """
        Retrieve a list of top-rated anime sorted by average rating in descending order.

        Args:
            limit (int): Maximum number of top-rated anime to return.
            min_vote_count (int, optional): Minimum number of votes required for an anime to be included.

        Returns:
            dict: A dictionary containing the list of top-rated anime.
        """
        # 1. Input parameter validation
        if not isinstance(limit, int) or limit <= 0:
            raise ValueError("The 'limit' parameter must be a positive integer.")

        if min_vote_count is not None:
            if not isinstance(min_vote_count, int) or min_vote_count < 0:
                raise ValueError("The 'min_vote_count' parameter must be a non-negative integer.")

        # 2. Access the anime table from the database
        # Using getattr to safely access the table, defaulting to an empty dict if not found
        anime_table = getattr(self.db, "anime", {})
        if not anime_table:
            return {"top_rated_anime": []}

        # 3. Filter anime based on criteria
        eligible_anime = []
        for anime in anime_table.values():
            # Retrieve rating and vote count, treating None values as 0 for comparison
            avg_rating = anime.average_rating if anime.average_rating is not None else 0.0
            votes = anime.vote_count if anime.vote_count is not None else 0

            # Apply min_vote_count filter if provided
            if min_vote_count is not None and votes < min_vote_count:
                continue

            eligible_anime.append(anime)

        # 4. Sort the filtered anime list
        # Primary sort: average_rating (descending)
        # Secondary sort: vote_count (descending) to handle ties in rating
        sorted_anime = sorted(
            eligible_anime,
            key=lambda x: (x.average_rating if x.average_rating is not None else 0.0, 
                           x.vote_count if x.vote_count is not None else 0),
            reverse=True
        )

        # 5. Apply the limit to the sorted list
        top_results = sorted_anime[:limit]

        # 6. Format the output according to the tool schema
        formatted_top_rated = []
        for anime in top_results:
            formatted_top_rated.append({
                "anime_id": anime.anime_id,
                "title": anime.title,
                "average_rating": float(anime.average_rating) if anime.average_rating is not None else 0.0,
                "vote_count": int(anime.vote_count) if anime.vote_count is not None else 0
            })

        return {"top_rated_anime": formatted_top_rated}

    @is_tool()
    def get_tv_series_crew_list(self, content_id: str):
            # Access the database tables for crew members and persons
            # The database tables are stored in self.db as dictionaries where keys are IDs
            crew_member_table = getattr(self.db, 'crew_member', {})
            person_table = getattr(self.db, 'person', {})

            # Filter the crew_member table for entries matching the requested content_id
            # content_id is a unique identifier, so we use exact matching
            # We iterate over the values of the crew_member dictionary
            matching_crew_entries = [
                member for member in crew_member_table.values()
                if member.content_id == content_id
            ]

            # The tool schema specifies that the TV series ID must exist in the database (pre-condition)
            # If no crew records are found for the provided content_id, we raise a KeyError as defined in the schema
            if not matching_crew_entries:
                raise KeyError(f"TV series ID '{content_id}' not found in the crew database.")

            # Construct the final crew list by combining crew member data with person names
            crew_list = []
            for member in matching_crew_entries:
                # Look up the person's name using the person_id from the person table
                person_info = person_table.get(member.person_id)

                # Extract the name from the person object. If person data is missing, use "Unknown"
                crew_name = person_info.name if person_info else "Unknown"

                # Append the crew member details to the result list
                # We handle cases where department might be None by defaulting to an empty string
                crew_list.append({
                    "crew_name": crew_name,
                    "role": member.role,
                    "department": member.department if member.department is not None else ""
                })

            # Return the result in the format specified by the tool schema returns definition
            return {"crew": crew_list}

    @is_tool()
    def get_tv_series_awards_list(self, content_id: str):
        # Access the 'award' table from the database
        # The database is accessed via self.db, and we use getattr for safety
        award_table = getattr(self.db, "award", {})

        # Check if the award_table is None (it might be initialized as None in the DB class)
        if award_table is None:
            award_table = {}

        # Initialize a list to store the awards and nominations for the specified TV series
        tv_series_awards = []

        # Iterate through all award records in the database
        # Since the award table is a dictionary (key: award_id, value: Award object), we iterate over values
        for award_entry in award_table.values():
            # Perform an exact match on content_id as it is a specific identifier
            if award_entry.content_id == content_id:
                # Create a dictionary for the award entry matching the tool schema's return structure
                # Award object attributes: award_name, category, result, year
                award_data = {
                    "award_name": award_entry.award_name,
                    "category": award_entry.category,
                    "result": award_entry.result,
                    "year": award_entry.year
                }
                tv_series_awards.append(award_data)

        # The tool schema specifies that the method should return an object with an 'awards' property
        # If no awards are found for the content_id, an empty list is returned
        return {"awards": tv_series_awards}

    @is_tool()
    def get_tv_series_details(self, content_id: str):
        """
        Retrieve complete metadata for a specific TV series including title, plot synopsis, cast, 
        air dates, number of seasons and episodes, genres, and ratings.
        """
        # Access the tv_series table from the database
        tv_series_table = getattr(self.db, "tv_series", {})
        if not tv_series_table or content_id not in tv_series_table:
            raise KeyError(f"TV series with ID '{content_id}' not found in the database.")

        # Retrieve the TV series object
        series = tv_series_table[content_id]

        # Initialize the return dictionary with basic metadata
        # Dates are formatted as 'yyyy-mm-dd' as per requirements
        result = {
            "series_id": series.series_id,
            "title": series.title,
            "plot_synopsis": series.plot_synopsis,
            "first_air_date": series.release_date.strftime("%Y-%m-%d") if series.release_date else None,
            "last_air_date": series.last_air_date.strftime("%Y-%m-%d") if series.last_air_date else None,
            "number_of_seasons": series.number_of_seasons,
            "number_of_episodes": series.episode_count,
            "average_rating": series.average_rating,
            "genres": [],
            "cast": []
        }

        # Retrieve genres associated with the TV series
        content_genre_table = getattr(self.db, "content_genre", {})
        if content_genre_table:
            # Since the Genre master table is not provided in the database schema, 
            # we return the genre_id as the genre name
            genres = [
                cg.genre_id for cg in content_genre_table.values() 
                if cg.content_id == content_id
            ]
            result["genres"] = genres

        # Retrieve cast members associated with the TV series
        cast_member_table = getattr(self.db, "cast_member", {})
        if cast_member_table:
            # Filter cast members for this specific content_id
            # Note: Since the Person table is not provided, person_id is used as actor_name
            cast_list = []
            for cast in cast_member_table.values():
                if cast.content_id == content_id:
                    cast_list.append({
                        "actor_name": cast.person_id,
                        "character_name": cast.character_name
                    })
            result["cast"] = cast_list

        return result

    @is_tool()
    def get_tv_series_alternative_titles(self, content_id: str):
        """
        Retrieve alternative titles and translations for a specific TV series in different languages and regions.

        Args:
            content_id (str): Unique identifier for the TV series.

        Returns:
            dict: A dictionary containing a list of alternative titles, each with title, language, and region.

        Raises:
            KeyError: If no alternative titles are found for the given content_id or if the database is inaccessible.
        """
        # Access the database instance from the class
        db = self.db

        # Retrieve the 'alternative_title' table from the database
        # According to the database definition, alternative_title is a Dict[str, AlternativeTitle]
        alt_titles_table = getattr(db, 'alternative_title', None)

        if alt_titles_table is None:
            # If the table is not found or not initialized, we raise a KeyError as per schema requirements
            raise KeyError("The alternative titles database table is currently unavailable.")

        # Initialize the list to collect matching titles
        alternative_titles_list = []

        # Iterate through the alternative titles stored in the database
        # Since content_id is a unique identifier (ID), we use exact matching (==)
        # rather than fuzzy matching, which is reserved for natural language text fields.
        for alt_id, alt_obj in alt_titles_table.items():
            if alt_obj.content_id == content_id:
                # Append the formatted title information to our results list
                alternative_titles_list.append({
                    "title": alt_obj.title,
                    "language": alt_obj.language if alt_obj.language else "Unknown",
                    "region": alt_obj.region if alt_obj.region else "Unknown"
                })

        # The tool schema specifies a pre-condition that the TV series ID must exist in the database.
        # It also specifies 'raises: KeyError'. If no entries are found for the provided content_id,
        # we assume the ID does not exist in the context of alternative titles and raise KeyError.
        if not alternative_titles_list:
            raise KeyError(f"No alternative titles found for TV series ID '{content_id}'.")

        # Return the results in the format defined by the tool schema's 'returns' field
        return {"alternative_titles": alternative_titles_list}

    @is_tool()
    def filter_tv_series_by_rating_range(self, min_rating: float, max_rating: float, limit: int = None):
        """
        Filter and retrieve TV series with ratings within a specified range.

        Args:
            min_rating (float): Minimum rating threshold (inclusive).
            max_rating (float): Maximum rating threshold (inclusive).
            limit (int, optional): Maximum number of results to return.

        Returns:
            dict: A dictionary containing a list of TV series results.

        Raises:
            ValueError: If pre-conditions are not met (e.g., min_rating > max_rating or ratings out of 0-10 range).
        """
        # 1. Validate pre-conditions
        if min_rating > max_rating:
            raise ValueError("Minimum rating must be less than or equal to maximum rating.")

        if not (0 <= min_rating <= 10) or not (0 <= max_rating <= 10):
            raise ValueError("Ratings must be between 0 and 10.")

        # 2. Access the database
        db = self.db
        tv_series_table = getattr(db, 'tv_series', None)

        if tv_series_table is None:
            return {"results": []}

        # 3. Filter TV series based on the rating range
        filtered_results = []

        # Iterate through all TV series in the database
        for series_id, series_obj in tv_series_table.items():
            # Handle cases where average_rating might be None
            rating = series_obj.average_rating
            if rating is not None:
                if min_rating <= rating <= max_rating:
                    filtered_results.append({
                        "series_id": series_obj.series_id,
                        "title": series_obj.title,
                        "average_rating": rating
                    })

        # 4. Apply limit if provided
        if limit is not None and limit > 0:
            # Optionally sort by rating descending for better quality results if not specified, 
            # but here we follow the basic requirement and just slice.
            filtered_results = filtered_results[:limit]

        return {"results": filtered_results}

    @is_tool()
    def get_movie_language_list(self, content_id: str):
        # Access the content_language table from the database.
        # According to the tool schema, the only related database is 'content_language'.
        # We use getattr to safely retrieve the table, defaulting to an empty dictionary if it doesn't exist.
        content_language_table = getattr(self.db, "content_language", {})

        # Initialize lists to store the available audio and subtitle languages.
        audio_languages = []
        subtitle_languages = []

        # Flag to track if we have found any records for the specified movie ID.
        movie_exists = False

        # Iterate through all entries in the content_language table.
        # The table is a dictionary where the keys are unique identifiers and values are ContentLanguage objects.
        if content_language_table:
            for entry in content_language_table.values():
                # Check if the record's content_id matches the requested content_id.
                # We use exact matching here as content_id is a unique identifier, not a natural language text field.
                if entry.content_id == content_id:
                    movie_exists = True

                    # Check the type of language track (audio or subtitle) and add to the respective list.
                    # We ensure the lists contain unique values to avoid duplicates.
                    if entry.language_type == "audio":
                        if entry.language not in audio_languages:
                            audio_languages.append(entry.language)
                    elif entry.language_type == "subtitle":
                        if entry.language not in subtitle_languages:
                            subtitle_languages.append(entry.language)

        # The tool schema pre-condition states that the Movie ID must exist in the database.
        # Additionally, the schema specifies that a KeyError may be raised.
        # If no records were found for the given content_id, we raise a KeyError.
        if not movie_exists:
            raise KeyError(f"Movie with ID '{content_id}' was not found in the language records.")

        # Return the dictionary containing the lists of audio and subtitle languages as required by the schema.
        return {
            "audio_languages": audio_languages,
            "subtitle_languages": subtitle_languages
        }

    @is_tool()
    def get_movie_images_list(self, content_id: str, image_type: Literal["poster", "backdrop", "still", "all"] = "all"):
        """
        Retrieve a list of promotional images, posters, and stills available for a specific movie.
        """
        # Access the entertainment_media_query database
        db = self.db

        # Retrieve the media_asset table from the database
        media_assets = getattr(db, "media_asset", None)

        # Pre-condition: Movie ID must exist in the database.
        # If the table is missing or empty, the movie ID cannot exist.
        if not media_assets:
            raise KeyError(f"Movie with ID '{content_id}' not found.")

        # Filter assets belonging to the specified content_id
        # content_id is an identifier, so we use exact matching as per instructions
        movie_assets = [asset for asset in media_assets.values() if asset.content_id == content_id]

        # If no assets are found for the given content_id, raise KeyError as required by tool schema
        if not movie_assets:
            raise KeyError(f"Movie with ID '{content_id}' not found.")

        # Safety protection for enum parameter: Ensure the provided image_type is valid
        valid_options = ["poster", "backdrop", "still", "all"]
        if image_type not in valid_options:
            raise ValueError(f"Invalid image_type '{image_type}'. Expected one of {valid_options}")

        # Define valid promotional image types to be returned (excluding trailers and other non-image assets)
        promo_types = ["poster", "backdrop", "still"]

        images_to_return = []

        for asset in movie_assets:
            # Determine if this asset matches the requested type and is a promotional image
            is_match = False

            if image_type == "all":
                # If 'all' is requested, include all standard promotional image types
                if asset.asset_type in promo_types:
                    is_match = True
            else:
                # If a specific type is requested, perform exact match
                # The safety check above ensures image_type is one of the promo_types if not "all"
                if asset.asset_type == image_type:
                    is_match = True

            if is_match:
                # Format the asset data for the return list according to the tool schema
                images_to_return.append({
                    "image_url": asset.url,
                    "image_type": asset.asset_type,
                    "width": asset.width,
                    "height": asset.height
                })

        # Return the dictionary containing the list of images
        return {"images": images_to_return}

    @is_tool()
    def get_anime_details(self, content_id: str):
        """
        Retrieve complete metadata for a specific anime including title, plot synopsis, 
        voice cast, air dates, number of episodes, genres, ratings, and studio information.
        """
        # Access the database instance
        db = self.db

        # Access the anime table
        # The database stores anime records in a dictionary keyed by anime_id
        anime_table = getattr(db, "anime", {})
        if not anime_table or content_id not in anime_table:
            raise KeyError(f"Anime with ID '{content_id}' not found.")

        anime_obj = anime_table[content_id]

        # Retrieve genres associated with this anime from the content_genre junction table
        # Since a separate Genre table is not provided in the schema, we use genre_id as the genre name
        genres = []
        content_genre_table = getattr(db, "content_genre", {})
        if content_genre_table:
            for genre_link in content_genre_table.values():
                if genre_link.content_id == content_id:
                    genres.append(genre_link.genre_id)

        # Retrieve voice cast information from the cast_member table
        # Filter for members associated with this content_id who have the role 'voice_actor'
        voice_cast = []
        cast_member_table = getattr(db, "cast_member", {})
        if cast_member_table:
            for member in cast_member_table.values():
                if member.content_id == content_id and member.role == "voice_actor":
                    voice_cast.append({
                        "voice_actor_name": member.person_id,  # Using person_id as the name proxy as no Person table is provided
                        "character_name": member.character_name
                    })

        # Format air dates to 'yyyy-mm-dd' strings as required by the tool schema
        # The database stores these as datetime objects
        first_air_date_str = None
        if anime_obj.release_date:
            first_air_date_str = anime_obj.release_date.strftime("%Y-%m-%d")

        last_air_date_str = None
        if anime_obj.last_air_date:
            last_air_date_str = anime_obj.last_air_date.strftime("%Y-%m-%d")

        # Construct the final metadata dictionary matching the returns schema
        # Note: 'studio' maps to 'company_name' in the Anime schema
        # 'number_of_episodes' maps to 'episode_count' in the Anime schema
        anime_details = {
            "anime_id": anime_obj.anime_id,
            "title": anime_obj.title,
            "plot_synopsis": anime_obj.plot_synopsis,
            "first_air_date": first_air_date_str,
            "last_air_date": last_air_date_str,
            "number_of_episodes": anime_obj.episode_count,
            "genres": genres,
            "average_rating": anime_obj.average_rating,
            "studio": anime_obj.company_name,
            "voice_cast": voice_cast
        }

        return anime_details

    @is_tool()
    def get_anime_staff_list(self, content_id: str):
        # Retrieve the staff list (crew members) and persons tables from the database
        # self.db is the instance of EntertainmentMediaQueryDB
        crew_table = getattr(self.db, 'crew_member', {})
        person_table = getattr(self.db, 'person', {})

        # Ensure tables are treated as dictionaries even if they are None in the DB
        if crew_table is None:
            crew_table = {}
        if person_table is None:
            person_table = {}

        staff_results = []
        content_id_found = False

        # Iterate through all crew member records to find those associated with the content_id
        # content_id is an identifier, so we use exact matching (==)
        for crew_id, crew_member in crew_table.items():
            if crew_member.content_id == content_id:
                content_id_found = True

                # Retrieve the corresponding person details using person_id from the crew record
                person_id = crew_member.person_id
                person_record = person_table.get(person_id)

                if person_record:
                    # Map the database fields to the tool's required return format:
                    # - staff_name: The name of the person from the Person table
                    # - role: The role of the staff member (e.g., Director) from the CrewMember table
                    # - position: The department or specific position (e.g., Series Director)
                    #   Since position isn't a direct field in CrewMember, we use the 'department' field
                    staff_results.append({
                        "staff_name": person_record.name,
                        "role": crew_member.role,
                        "position": crew_member.department if crew_member.department else ""
                    })

        # Pre-condition: Anime ID must exist in the database. 
        # If no records were found for the provided content_id, raise a KeyError.
        if not content_id_found:
            raise KeyError(f"Anime ID '{content_id}' not found in the database.")

        # Return the collected staff list in the required format
        return {"crew": staff_results}

    @is_tool()
    def get_anime_air_schedule(self, start_date: str, end_date: str, limit: int = 100):
        """
        Retrieve anime scheduled to air within a specified date range.
        """
        from datetime import datetime

        # 1. Parse date strings into datetime objects
        # This helper handles both full datetime strings and date-only strings as per requirements
        def parse_dt(dt_str: str) -> datetime:
            for fmt in ("%Y-%m-%d %H:%M:%S", "%Y-%m-%d"):
                try:
                    return datetime.strptime(dt_str.strip(), fmt)
                except ValueError:
                    continue
            raise ValueError(f"Time data '{dt_str}' does not match format 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd'")

        try:
            start_dt = parse_dt(start_date)
            end_dt = parse_dt(end_date)
        except ValueError as e:
            raise ValueError(str(e))

        # If end_date is provided as a date only (length 10), we make it inclusive of the whole day
        # to ensure episodes airing later that day are captured.
        if len(end_date.strip()) == 10:
            end_dt = end_dt.replace(hour=23, minute=59, second=59)

        # 2. Validate pre-condition: Start date must be before or equal to end date
        if start_dt > end_dt:
            raise ValueError("Start date must be before or equal to end date")

        # 3. Access databases
        db = self.db
        episodes_dict = getattr(db, "episode", {}) or {}
        animes_dict = getattr(db, "anime", {}) or {}

        if not episodes_dict or not animes_dict:
            return {"scheduled_anime": []}

        # 4. Filter episodes airing within the specified range
        scheduled_list = []

        # We iterate through episodes to find those within the date range
        for episode in episodes_dict.values():
            if episode.release_date:
                # Standardize comparison using datetime objects
                if start_dt <= episode.release_date <= end_dt:
                    # Find the corresponding anime details using content_id (which references anime_id)
                    anime_id = episode.content_id
                    anime_info = animes_dict.get(anime_id)

                    if anime_info:
                        # Format the air_date string: use full format if time is present, otherwise date-only
                        if episode.release_date.hour == 0 and episode.release_date.minute == 0 and episode.release_date.second == 0:
                            air_date_str = episode.release_date.strftime("%Y-%m-%d")
                        else:
                            air_date_str = episode.release_date.strftime("%Y-%m-%d %H:%M:%S")

                        scheduled_list.append({
                            "anime_id": anime_id,
                            "title": anime_info.title,
                            "air_date": air_date_str
                        })

        # 5. Sort the results by air date to provide a chronological schedule
        scheduled_list.sort(key=lambda x: x["air_date"])

        # 6. Apply the limit if provided
        if limit and limit > 0:
            scheduled_list = scheduled_list[:limit]

        return {"scheduled_anime": scheduled_list}

    @is_tool()
    def get_anime_alternative_titles(self, content_id: str):
        # Access the entertainment_media_query database
        db = self.db

        # Retrieve the alternative_title table from the database
        # This table stores titles in various languages and regions linked by content_id
        alt_titles_table = getattr(db, 'alternative_title', None)

        # Check if the table exists in the database
        if alt_titles_table is None:
            raise KeyError("The alternative_title database table is not available.")

        alternative_titles_list = []

        # Iterate through all alternative title records in the table
        # We use exact matching for content_id as it is a unique identifier
        for alt_title_record in alt_titles_table.values():
            if alt_title_record.content_id == content_id:
                # Construct the title information object as per the returns schema
                title_info = {
                    "title": alt_title_record.title,
                    "language": alt_title_record.language,
                    "region": alt_title_record.region
                }
                alternative_titles_list.append(title_info)

        # The pre-condition states the Anime ID must exist in the database.
        # If no records are found for the given content_id, we raise a KeyError as defined in the schema.
        if not alternative_titles_list:
            raise KeyError(f"No alternative titles found for anime ID: {content_id}")

        # Return the list of alternative titles wrapped in the expected dictionary structure
        return {"alternative_titles": alternative_titles_list}

    @is_tool()
    def get_anime_awards_list(self, content_id: str):
        # Obtain the award database table
        # Using getattr to safely access the 'award' table from the database instance
        award_table = getattr(self.db, 'award', None)

        # Check if the award table exists or is empty
        if not award_table:
            # Based on the tool schema's raises and pre-condition requirement,
            # we raise a KeyError if the data source is unavailable or the ID is not found.
            raise KeyError(f"Anime ID '{content_id}' not found in the database.")

        # Filter the awards based on the provided content_id
        # Note: IDs are treated as exact identifiers, so fuzzy matching is not applied here.
        matching_awards = []
        for award_entry in award_table.values():
            if award_entry.content_id == content_id:
                # Format the award data according to the return schema requirements
                matching_awards.append({
                    "award_name": award_entry.award_name,
                    "category": award_entry.category,
                    "result": award_entry.result,
                    "year": award_entry.year
                })

        # The pre-condition states the Anime ID must exist in the database.
        # If no awards were found for this content_id, we interpret this as the ID not existing 
        # in the context of the awards database.
        if not matching_awards:
            raise KeyError(f"No award records found for anime ID: {content_id}")

        # Return the list of awards wrapped in the required object structure
        return {"awards": matching_awards}

    @is_tool()
    def get_tv_series_language_list(self, content_id: str):
        """
        Retrieve the list of available audio languages and subtitle languages for a specific TV series.
        """
        # Access the database instance
        db = self.db

        # Access the content_language table which stores language information for media content
        # The related_databases field indicates we should use 'content_language'
        content_language_table = getattr(db, 'content_language', None)

        # If the table is missing from the database, the ID cannot be found
        if content_language_table is None:
            raise KeyError(f"The content_language database table is not available. Could not find ID '{content_id}'.")

        audio_languages = set()
        subtitle_languages = set()
        content_found = False

        # Iterate through all entries in the content_language table to find matches for the content_id
        for entry in content_language_table.values():
            # identifiers like content_id must use exact matching
            if entry.content_id == content_id:
                content_found = True

                # Categorize the language based on the language_type (audio or subtitle)
                # The schema defines language_type as Literal["audio", "subtitle"]
                if entry.language_type == "audio":
                    audio_languages.add(entry.language)
                elif entry.language_type == "subtitle":
                    subtitle_languages.add(entry.language)

        # According to the tool schema, the pre-condition is that the TV series ID must exist.
        # If no records were found matching the content_id, raise a KeyError.
        if not content_found:
            raise KeyError(f"TV series with ID '{content_id}' was not found in the database.")

        # Return the collected languages as lists, sorted for deterministic output
        return {
            "audio_languages": sorted(list(audio_languages)),
            "subtitle_languages": sorted(list(subtitle_languages))
        }

    @is_tool()
    def get_tv_series_season_list(self, content_id: str):
        """
        Retrieve a list of all seasons for a specific TV series, including season numbers and episode counts.

        Args:
            content_id (str): Unique identifier for the TV series (corresponds to series_id in the season table).

        Returns:
            dict: A dictionary containing a list of seasons, where each season includes its number,
                  episode count, and air date.

        Raises:
            KeyError: Raised if the TV series ID is not found in the database or no seasons are available.
        """
        # Access the 'season' table from the database
        # The database instance is available via self.db
        season_table = getattr(self.db, "season", None)

        # If the season table doesn't exist, we cannot fulfill the request
        if season_table is None:
            raise KeyError("The 'season' database table is not available.")

        seasons_data = []

        # Iterate through the season records to find matches for the given series_id
        # Note: content_id is an identifier, so we use exact matching (==)
        for season in season_table.values():
            if season.series_id == content_id:
                # Format the release_date to "yyyy-mm-dd" string format as required
                # The prompt specifies "yyyy-mm-dd" for date-only values
                air_date_str = ""
                if season.release_date:
                    # release_date is a datetime.datetime object in the database
                    air_date_str = season.release_date.strftime("%Y-%m-%d")

                # Construct the season object as defined in the tool schema returns
                seasons_data.append({
                    "season_number": season.season_number,
                    "episode_count": season.episode_count if season.episode_count is not None else 0,
                    "air_date": air_date_str
                })

        # The tool schema specifies KeyError and the pre-condition states the TV series ID must exist.
        # If no records are found matching the content_id, we raise KeyError.
        if not seasons_data:
            raise KeyError(f"TV series ID '{content_id}' not found in the database.")

        # Sort the list by season_number to provide a structured and logical result
        seasons_data.sort(key=lambda x: x["season_number"])

        return {"seasons": seasons_data}

    @is_tool()
    def get_tv_series_images_list(self, content_id: str, image_type: Literal["poster", "backdrop", "still", "all"] = "all"):
            """
            Retrieve a list of promotional images, posters, and stills available for a specific TV series.
            """
            # Access the media_asset table from the database
            media_assets = getattr(self.db, "media_asset", None)

            if not media_assets:
                raise KeyError(f"Media asset database is empty or not found.")

            # Identify all assets associated with the given content_id
            # We use exact matching for IDs as per instructions
            content_assets = [asset for asset in media_assets.values() if asset.content_id == content_id]

            # Pre-condition check: TV series ID must exist in the database (at least one asset should exist)
            if not content_assets:
                raise KeyError(f"No media assets found for TV series ID: {content_id}")

            # Define the valid image types based on the tool's focus (posters, backdrops, stills)
            valid_image_types = ["poster", "backdrop", "still"]

            results = []
            for asset in content_assets:
                # Determine if the asset matches the requested image_type
                match_type = False

                # If "all" is requested, we include all standard image types
                if image_type == "all":
                    if asset.asset_type in valid_image_types:
                        match_type = True
                # Otherwise, check for an exact match with the requested type
                elif asset.asset_type == image_type:
                    match_type = True

                if match_type:
                    # Append the image details to the results list
                    results.append({
                        "image_url": asset.url,
                        "image_type": asset.asset_type,
                        "width": asset.width if asset.width is not None else 0,
                        "height": asset.height if asset.height is not None else 0
                    })

            return {"images": results}

    @is_tool()
    def search_movies_by_actor(self, actor_name: str, limit: int = 20):
        # Validate the input: actor_name must be a non-empty string as per pre-condition
        if not actor_name or not isinstance(actor_name, str):
            raise ValueError("Actor name must be a non-empty string")

        # Access the database tables safely using only those defined in related_databases
        db = self.db
        person_table = getattr(db, "person", {}) or {}
        cast_table = getattr(db, "cast_member", {}) or {}
        movie_table = getattr(db, "movie", {}) or {}

        # 1. Identify the actor(s) using fuzzy matching on the person table to improve robustness
        # We create a mapping of person_id to their full name for the search
        person_choices = {p_id: p.name for p_id, p in person_table.items()}

        # Use process.extract to handle both partial and slightly misspelled names
        # We use a limit of 50 to ensure we capture relevant actors even for common names
        matches = process.extract(actor_name, person_choices, limit=50)

        matched_person_ids = set()
        for name, score, p_id in matches:
            # Filter matches based on a similarity threshold or if the query is a clear substring
            if score >= 80 or actor_name.lower() in name.lower():
                matched_person_ids.add(p_id)

        # If no matching persons are found, return an empty list of movies
        if not matched_person_ids:
            return {"movies": []}

        # 2. Retrieve movie appearances for the matched persons
        # Efficiency: Filter the cast members for the matched persons first before sorting
        relevant_cast = [
            cast for cast in cast_table.values() 
            if cast.person_id in matched_person_ids and cast.role in ["actor", "voice_actor"]
        ]

        # Sort by billing order to prioritize lead roles (lower billing order usually means more prominent)
        relevant_cast.sort(key=lambda x: x.billing_order if x.billing_order is not None else 9999)

        movies_found = []
        seen_entries = set()

        for cast in relevant_cast:
            # Check if the content_id exists in the movie table
            movie = movie_table.get(cast.content_id)
            if movie:
                # Use a unique key (movie_id + character) to avoid duplicate entries for the same role
                unique_key = (movie.movie_id, cast.character_name)
                if unique_key not in seen_entries:
                    movie_info = {
                        "movie_id": movie.movie_id,
                        "title": movie.title,
                        "character_name": cast.character_name
                    }

                    # Extract the release year as an integer from the datetime object if available
                    # This aligns with the return schema requirement for release_year: integer
                    if movie.release_date:
                        movie_info["release_year"] = movie.release_date.year

                    movies_found.append(movie_info)
                    seen_entries.add(unique_key)

            # Check if we have reached the user-specified limit for the number of movies
            if len(movies_found) >= limit:
                break

        return {"movies": movies_found}

    @is_tool()
    def filter_anime_by_rating_range(self, min_rating: float, max_rating: float, limit: int = None) -> dict:
        """
        Filters and retrieves anime with ratings within a specified range from the database.

        Args:
            min_rating (float): Minimum rating threshold (inclusive).
            max_rating (float): Maximum rating threshold (inclusive).
            limit (int, optional): Maximum number of results to return.

        Returns:
            dict: A dictionary containing a list of anime matching the criteria.
        """
        # Validation of the rating range as per pre-condition
        if min_rating < 0 or min_rating > 10:
            raise ValueError("min_rating must be between 0 and 10.")
        if max_rating < 0 or max_rating > 10:
            raise ValueError("max_rating must be between 0 and 10.")
        if min_rating > max_rating:
            raise ValueError("min_rating must be less than or equal to max_rating.")

        # Access the anime table from the database
        anime_table = getattr(self.db, "anime", {})
        if not anime_table:
            return {"results": []}

        filtered_results = []

        # Iterate through all anime records
        for anime_id, anime_obj in anime_table.items():
            # Check if the average_rating exists and falls within the specified range
            # Note: average_rating is Optional[float] in the schema
            rating = anime_obj.average_rating
            if rating is not None and min_rating <= rating <= max_rating:
                filtered_results.append({
                    "anime_id": anime_obj.anime_id,
                    "title": anime_obj.title,
                    "average_rating": rating
                })

        # Sort results by rating descending by default for better utility, 
        # though the schema doesn't strictly require sorting.
        filtered_results.sort(key=lambda x: x["average_rating"], reverse=True)

        # Apply limit if provided
        if limit is not None and limit > 0:
            filtered_results = filtered_results[:limit]

        return {"results": filtered_results}

    @is_tool()
    def get_related_movies(self, content_id: str, limit: int = 10):
        """
        Retrieve a list of movies related to a specified movie based on similarity in genre, cast, crew, or themes.
        """
        # Access the database
        db = self.db
        movie_table = getattr(db, "movie") or {}

        # Pre-condition: Movie ID must exist in the database
        if content_id not in movie_table:
            raise KeyError(f"Movie with ID '{content_id}' does not exist in the database.")

        target_movie = movie_table[content_id]

        # Retrieve related tables for comparison
        content_genre_table = getattr(db, "content_genre") or {}
        cast_member_table = getattr(db, "cast_member") or {}
        crew_member_table = getattr(db, "crew_member") or {}

        # Optimize data retrieval by pre-grouping attributes by content_id
        # This prevents O(N*M) lookups inside the comparison loop
        genres_by_content = {}
        for cg in content_genre_table.values():
            if cg.content_id not in genres_by_content:
                genres_by_content[cg.content_id] = set()
            genres_by_content[cg.content_id].add(cg.genre_id)

        cast_by_content = {}
        for cm in cast_member_table.values():
            if cm.content_id not in cast_by_content:
                cast_by_content[cm.content_id] = set()
            cast_by_content[cm.content_id].add(cm.person_id)

        crew_by_content = {}
        for cr in crew_member_table.values():
            if cr.content_id not in crew_by_content:
                crew_by_content[cr.content_id] = set()
            crew_by_content[cr.content_id].add(cr.person_id)

        # Extract target movie's attributes for comparison
        target_genres = genres_by_content.get(content_id, set())
        target_cast = cast_by_content.get(content_id, set())
        target_crew = crew_by_content.get(content_id, set())
        target_synopsis = target_movie.plot_synopsis or ""

        related_movies_list = []

        # Iterate through all movies to calculate similarity scores
        for mid, movie in movie_table.items():
            # Skip the reference movie itself
            if mid == content_id:
                continue

            # 1. Genre Similarity (using Jaccard Index)
            m_genres = genres_by_content.get(mid, set())
            genre_score = 0.0
            if target_genres or m_genres:
                genre_score = len(target_genres & m_genres) / len(target_genres | m_genres)

            # 2. Cast Similarity (shared actors)
            m_cast = cast_by_content.get(mid, set())
            cast_score = 0.0
            if target_cast or m_cast:
                cast_score = len(target_cast & m_cast) / len(target_cast | m_cast)

            # 3. Crew Similarity (shared directors, writers, etc.)
            m_crew = crew_by_content.get(mid, set())
            crew_score = 0.0
            if target_crew or m_crew:
                crew_score = len(target_crew & m_crew) / len(target_crew | m_crew)

            # 4. Theme Similarity (comparing plot synopses using fuzzy matching)
            m_synopsis = movie.plot_synopsis or ""
            theme_score = 0.0
            if target_synopsis and m_synopsis:
                # token_set_ratio is robust against different word orders and partial matches
                theme_score = fuzz.token_set_ratio(target_synopsis, m_synopsis) / 100.0

            # Calculate an aggregate similarity score using weighted averages
            # Weights: Genre (40%), Themes/Synopsis (30%), Cast (15%), Crew (15%)
            # These weights reflect standard recommendation logic where genre and content are primary signals
            similarity_score = (
                (genre_score * 0.4) + 
                (theme_score * 0.3) + 
                (cast_score * 0.15) + 
                (crew_score * 0.15)
            )

            # Only include movies with some level of similarity
            if similarity_score > 0:
                related_movies_list.append({
                    "movie_id": mid,
                    "title": movie.title,
                    "similarity_score": round(similarity_score, 3)
                })

        # Sort the list by similarity_score in descending order (highest similarity first)
        related_movies_list.sort(key=lambda x: x["similarity_score"], reverse=True)

        # Return the results limited by the specified parameter
        return {
            "related_movies": related_movies_list[:limit]
        }

    @is_tool()
    def search_tv_series_by_actor(self, actor_name: str, limit: int = None):
        """
        Search for TV series featuring a specific actor by searching the actor's name.
        Utilizes fuzzy matching to handle partial names or typos.
        """
        # Import necessary for type checking and date processing
        from datetime import datetime

        # Basic input validation
        if not actor_name or not isinstance(actor_name, str):
            raise ValueError("Actor name must be a non-empty string.")

        # Access the database tables
        db = self.db
        person_table = getattr(db, 'person', {})
        cast_table = getattr(db, 'cast_member', {})
        series_table = getattr(db, 'tv_series', {})

        # If any required table is missing or empty, return an empty list
        if not person_table or not cast_table or not series_table:
            return {"tv_series": []}

        # Step 1: Use fuzzy matching to find individuals in the person table matching actor_name
        # Mapping person_id to their full name for the fuzzy search
        person_choices = {pid: p.name for pid, p in person_table.items()}

        # Extract matches. We use a threshold of 80 to ensure relevance.
        # process.extract returns a list of (name, score, person_id)
        matches = process.extract(actor_name, person_choices, limit=20)
        matched_person_ids = [m[2] for m in matches if m[1] >= 80]

        if not matched_person_ids:
            return {"tv_series": []}

        results = []
        # Use a set to track unique (series_id, character_name) combinations to avoid duplicates
        seen_entries = set()

        # Step 2: Iterate through matched persons and find their acting roles in TV series
        for pid in matched_person_ids:
            # Filter cast_member entries for this person where the role is 'actor'
            for cast_id, cast in cast_table.items():
                if cast.person_id == pid and cast.role == "actor":
                    series_id = cast.content_id

                    # Step 3: Verify the content_id exists in the tv_series table 
                    # (since cast_member might also contain movie or anime references)
                    if series_id in series_table:
                        series = series_table[series_id]

                        # Create a unique key for the result entry
                        entry_key = (series_id, cast.character_name)
                        if entry_key not in seen_entries:
                            # Extract the year from release_date if it exists
                            first_air_year = None
                            if series.release_date and isinstance(series.release_date, datetime):
                                first_air_year = series.release_date.year

                            results.append({
                                "series_id": series.series_id,
                                "title": series.title,
                                "character_name": cast.character_name or "Unknown",
                                "first_air_year": first_air_year
                            })
                            seen_entries.add(entry_key)

        # Step 4: Apply the limit if provided
        if limit is not None and isinstance(limit, int) and limit > 0:
            results = results[:limit]

        return {"tv_series": results}

    @is_tool()
    def get_popular_movies_by_region(self, region_code: str, limit: int = 20):
        """
        Retrieve a list of popular movies in a specific geographic region based on viewership or popularity metrics.

        Args:
            region_code: ISO country code or region identifier (e.g., "US", "JP", "UK").
            limit: Maximum number of results to return. Defaults to 20.

        Returns:
            A dictionary containing a list of popular movies with their IDs, titles, and popularity scores.

        Raises:
            ValueError: If region_code is not provided or invalid.
        """
        # Validate input parameters
        if not region_code:
            raise ValueError("The 'region_code' parameter is required.")

        # Access the database tables
        # Use getattr and handle the case where the table might be None by using 'or {}'
        popularity_data = getattr(self.db, "content_popularity", {}) or {}
        movie_data = getattr(self.db, "movie", {}) or {}

        if not popularity_data:
            return {"popular_movies": []}

        # Filter content popularity records by the specified region
        # Since 'region_code' is an identifier/code, we use exact case-insensitive matching
        # per the instructions' exclusion for codes (❌ id, email, phone, code).
        regional_popularity = [
            item for item in popularity_data.values() 
            if item.region.strip().upper() == region_code.strip().upper()
        ]

        if not regional_popularity:
            return {"popular_movies": []}

        # Sort the records by popularity_score in descending order (highest first)
        sorted_popularity = sorted(
            regional_popularity, 
            key=lambda x: x.popularity_score, 
            reverse=True
        )

        # Apply the limit constraint
        if limit is not None and limit >= 0:
            sorted_popularity = sorted_popularity[:limit]

        popular_movies_list = []

        # Enrich the popularity data with movie titles from the movie table
        for entry in sorted_popularity:
            movie_id = entry.content_id
            # Look up the movie details in the movie table
            movie_item = movie_data.get(movie_id)

            # According to the return schema, title is required, so we only include movies found in the metadata
            if movie_item:
                popular_movies_list.append({
                    "movie_id": movie_id,
                    "title": movie_item.title,
                    "popularity_score": entry.popularity_score
                })

        return {"popular_movies": popular_movies_list}

    @is_tool()
    def get_anime_language_list(self, content_id: str):
        # Access the content_language table from the database
        # Strictly using only the 'content_language' table as specified in related_databases
        content_language_table = getattr(self.db, 'content_language', {}) or {}

        # Initialize sets to store unique languages found for the given content_id
        audio_languages = set()
        subtitle_languages = set()

        # Flag to track if the content_id exists in the language records
        id_found_in_languages = False

        # Iterate through all entries in the content_language table
        for entry in content_language_table.values():
            # Use exact matching for the content identifier (ID)
            if entry.content_id == content_id:
                id_found_in_languages = True
                # Categorize the language based on the language_type attribute ("audio" or "subtitle")
                if entry.language_type == "audio":
                    audio_languages.add(entry.language)
                elif entry.language_type == "subtitle":
                    subtitle_languages.add(entry.language)

        # If no records were found for the specified content_id, raise a KeyError
        # This satisfies the 'raises' requirement and the pre-condition check within allowed tables
        if not id_found_in_languages:
            raise KeyError(f"Anime ID '{content_id}' not found in the database.")

        # Return the results as a dictionary containing sorted lists of languages
        # Sorting ensures the output is deterministic
        return {
            "audio_languages": sorted(list(audio_languages)),
            "subtitle_languages": sorted(list(subtitle_languages))
        }

    @is_tool()
    def get_tv_series_episode_list(self, content_id: str, season_number: int):
        """
        Retrieve a list of all episodes for a specific season of a TV series, including episode titles, numbers, and air dates.

        Args:
            content_id: Unique identifier for the TV series.
            season_number: Season number to retrieve episodes for.

        Returns:
            A dictionary containing a list of episodes in the specified season.

        Raises:
            KeyError: If the TV series ID or the season number does not exist in the database.
        """
        from datetime import datetime

        # Access the episode table from the database as defined in related_databases
        db = self.db
        episode_table = getattr(db, "episode", None)

        if not episode_table:
            raise KeyError(f"No episode data found in the database. TV series ID '{content_id}' not found.")

        # Filter episodes by content_id first to check if the series exists (Pre-condition requirement)
        # IDs use exact matching.
        series_episodes = [ep for ep in episode_table.values() if ep.content_id == content_id]
        if not series_episodes:
            raise KeyError(f"TV series with ID '{content_id}' not found in the database.")

        # Filter by season_number to check if the season exists (Pre-condition requirement)
        season_episodes = [ep for ep in series_episodes if ep.season_number == season_number]
        if not season_episodes:
            raise KeyError(f"Season {season_number} for TV series '{content_id}' not found in the database.")

        episodes_list = []
        for ep in season_episodes:
            # Format the air date according to the requirement: "yyyy-mm-dd HH:MM:SS" or "yyyy-mm-dd"
            air_date_str = ""
            if ep.release_date:
                # Check if there is time information beyond midnight to decide on the format
                if ep.release_date.hour == 0 and ep.release_date.minute == 0 and ep.release_date.second == 0:
                    air_date_str = ep.release_date.strftime("%Y-%m-%d")
                else:
                    air_date_str = ep.release_date.strftime("%Y-%m-%d %H:%M:%S")

            # Map database fields to the tool schema return format
            episodes_list.append({
                "episode_number": ep.episode_number,
                "episode_title": ep.title if ep.title else "Untitled",
                "air_date": air_date_str,
                "runtime_minutes": ep.runtime if ep.runtime is not None else 0
            })

        # Sort the episodes by episode_number to provide an ordered list as expected for a TV series
        episodes_list.sort(key=lambda x: x["episode_number"])

        return {"episodes": episodes_list}

    @is_tool()
    def filter_movies_by_content_rating(self, content_rating: Literal["G", "PG", "PG-13", "R", "NC-17", "NR"], limit: int = None):
        # Access the movie table from the entertainment_media_query database
        # We use getattr to safely handle cases where the table might not be initialized
        movie_table = getattr(self.db, 'movie', None)

        # If the database or the movie table is empty, return an empty results list
        if not movie_table:
            return {"results": []}

        # Security protection: Validate the content_rating against the allowed enum values
        # Even though Literal is used in the signature, this provides runtime safety
        valid_ratings = ["G", "PG", "PG-13", "R", "NC-17", "NR"]
        if content_rating not in valid_ratings:
            raise ValueError(f"Invalid content_rating: '{content_rating}'. Must be one of {valid_ratings}")

        # Initialize the list to store matching movie results
        filtered_movies = []

        # Iterate through all movies in the database
        # Since content_rating is a categorical classification, we use exact matching (==)
        for movie in movie_table.values():
            # Check if the movie's content rating matches the requested classification
            # Note: movie.content_rating is defined as a Literal in the Movie schema
            if movie.content_rating == content_rating:
                # Construct the movie object according to the tool's return schema
                filtered_movies.append({
                    "movie_id": movie.movie_id,
                    "title": movie.title,
                    "content_rating": movie.content_rating
                })

        # Sort the results by title to ensure a consistent output order
        filtered_movies.sort(key=lambda x: x['title'])

        # Apply the result limit if provided
        if limit is not None:
            if limit < 0:
                raise ValueError("The 'limit' parameter must be a non-negative integer.")
            # Slice the list to return only the requested number of items
            filtered_movies = filtered_movies[:limit]

        # Return the results wrapped in the required dictionary structure
        return {"results": filtered_movies}

    @is_tool()
    def get_anime_images_list(self, content_id: str, image_type: Literal["poster", "backdrop", "character_art", "all"] = "all"):
        """
        Retrieve a list of promotional images, posters, and character art available for a specific anime.
        """
        # Safety protection: Check if the provided image_type is within the allowed enum values
        valid_image_types = ["poster", "backdrop", "character_art", "all"]
        if image_type not in valid_image_types:
            raise ValueError(f"Invalid image_type: '{image_type}'. Expected one of {valid_image_types}.")

        # Access the media_asset table from the database
        # Ensuring we handle the case where the table might be None
        media_asset_table = getattr(self.db, "media_asset", {}) or {}

        images_result = []
        content_id_exists = False

        # Define the specific image categories the tool is designed to handle
        # This helps in filtering the 'all' request to only relevant promotional images
        promotional_types = ["poster", "backdrop", "character_art"]

        # Iterate through all assets in the media_asset database
        for asset in media_asset_table.values():
            # Check for the specific anime ID (content_id)
            # Exact match is used as content_id is a unique identifier
            if asset.content_id == content_id:
                # Mark that the anime ID exists in the database (pre-condition check)
                content_id_exists = True

                # Determine if this specific asset matches the requested filter
                should_include = False
                if image_type == "all":
                    # If 'all' is requested, include assets that match the promotional categories
                    # We exclude 'trailer' (video) and potentially other non-promotional types like 'still'
                    if asset.asset_type in promotional_types:
                        should_include = True
                elif asset.asset_type == image_type:
                    # If a specific type is requested, perform an exact match
                    should_include = True

                if should_include:
                    # Construct the image dictionary according to the return schema
                    # Width and height are defaulted to 0 if they are None in the database
                    images_result.append({
                        "image_url": asset.url,
                        "image_type": asset.asset_type,
                        "width": asset.width if asset.width is not None else 0,
                        "height": asset.height if asset.height is not None else 0
                    })

        # Enforcement of the pre-condition: The Anime ID must exist in the database.
        # If no assets (including trailers) were found for this content_id, it is considered missing.
        if not content_id_exists:
            raise KeyError(f"Anime ID '{content_id}' not found in the media asset database.")

        # Return the results in the format specified by the tool schema
        return {"images": images_result}

    @is_tool()
    def search_tv_series_by_keyword(self, keyword: str, limit: int = 10):
            """
            Search for television series using a keyword that matches title, plot synopsis, or other text fields.
            Uses fuzzy matching to improve search robustness for natural language queries.
            """
            # 1. Pre-condition validation: Keyword must be a non-empty string
            if not keyword or not isinstance(keyword, str) or not keyword.strip():
                raise ValueError("Keyword must be a non-empty string")

            # 2. Access the tv_series database table
            tv_series_table = getattr(self.db, 'tv_series', None)
            if not tv_series_table:
                return {"results": []}

            # 3. Perform fuzzy matching across relevant text fields
            matches = []
            search_keyword = keyword.lower()

            for series in tv_series_table.values():
                # Calculate similarity scores for different fields
                # Title is the most important field
                title_score = fuzz.partial_ratio(search_keyword, series.title.lower())

                # Plot synopsis is secondary
                plot_score = 0
                if series.plot_synopsis:
                    plot_score = fuzz.partial_ratio(search_keyword, series.plot_synopsis.lower())

                # Determine the best match score for this series
                # We give a slight weight/priority to title matches
                best_score = max(title_score, plot_score)

                # Define a threshold for relevance (e.g., 60 out of 100)
                if best_score >= 60:
                    # Extract year from release_date (datetime object)
                    first_air_year = None
                    if series.release_date:
                        first_air_year = series.release_date.year

                    matches.append({
                        "score": best_score,
                        "series_id": series.series_id,
                        "title": series.title,
                        "first_air_year": first_air_year
                    })

            # 4. Sort matches by score in descending order (highest relevance first)
            matches.sort(key=lambda x: x["score"], reverse=True)

            # 5. Format the results according to the tool schema
            # We only need series_id, title, and first_air_year in the output
            formatted_results = []
            for m in matches[:limit]:
                result_item = {
                    "series_id": m["series_id"],
                    "title": m["title"],
                    "first_air_year": m["first_air_year"]
                }
                formatted_results.append(result_item)

            return {"results": formatted_results}

    @is_tool()
    def filter_movies_by_genre(
            self,
            genres: List[Literal["Action", "Adventure", "Animation", "Biography", "Comedy", "Crime", "Documentary", "Drama", "Family", "Fantasy", "Film-Noir", "History", "Horror", "Music", "Musical", "Mystery", "Romance", "Sci-Fi", "Sport", "Thriller", "War", "Western"]],
            match_all: bool = False,
            limit: int = 20
        ) -> Dict[str, Any]:
            """
            Filter and retrieve movies that belong to specified genres.
            """
            # Validate that genres list is not empty as per schema requirements
            if not genres:
                raise ValueError("The 'genres' parameter cannot be empty.")

            # Access database tables
            db = self.db
            movie_table = getattr(db, "movie", {})
            genre_table = getattr(db, "genre", {})
            content_genre_table = getattr(db, "content_genre", {})

            # 1. Map input genre names to database genre IDs
            # We use fuzzy matching to find the best match for each input genre in the database.
            db_genre_names = {g.genre_name: g.genre_id for g in genre_table.values()}
            id_to_name = {g.genre_id: g.genre_name for g in genre_table.values()}

            target_genre_ids = set()
            genre_choices = list(db_genre_names.keys())

            if not genre_choices:
                return {"results": []}

            for input_genre in genres:
                # Use fuzzy matching to find the closest genre name in the database
                match_result = process.extractOne(input_genre, genre_choices, scorer=fuzz.token_sort_ratio)
                if match_result and match_result[1] >= 80:
                    target_genre_ids.add(db_genre_names[match_result[0]])
                elif match_all:
                    # If match_all is True and an input genre is not found in the DB, no movie can match
                    return {"results": []}

            # If no valid genres were found in the database (and match_all was False)
            if not target_genre_ids:
                return {"results": []}

            # 2. Build a mapping of movie_id to their associated genre_ids
            # This allows us to efficiently check the genres of each movie.
            movie_to_genre_ids = {}
            for cg in content_genre_table.values():
                m_id = cg.content_id
                g_id = cg.genre_id
                if m_id not in movie_to_genre_ids:
                    movie_to_genre_ids[m_id] = set()
                movie_to_genre_ids[m_id].add(g_id)

            # 3. Filter movies based on the genre criteria
            filtered_results = []
            # Sort movie keys for consistent output order
            for movie_id in sorted(movie_table.keys()):
                movie_obj = movie_table[movie_id]
                movie_gids = movie_to_genre_ids.get(movie_id, set())

                is_match = False
                if match_all:
                    # Movie must contain ALL specified genres (target_genre_ids is a subset of movie_gids)
                    if target_genre_ids.issubset(movie_gids):
                        is_match = True
                else:
                    # Movie must contain AT LEAST ONE specified genre (intersection is not empty)
                    if not target_genre_ids.isdisjoint(movie_gids):
                        is_match = True

                if is_match:
                    # Retrieve the full list of genre names for the matching movie
                    movie_genre_names = [id_to_name[gid] for gid in movie_gids if gid in id_to_name]
                    filtered_results.append({
                        "movie_id": movie_id,
                        "title": movie_obj.title,
                        "genres": movie_genre_names
                    })

            # 4. Return the results limited by the specified limit
            return {"results": filtered_results[:limit]}

    @is_tool()
    def get_anime_episode_list(self, content_id: str):
        """
        Retrieve a list of all episodes for a specific anime, including episode titles, numbers, and air dates.

        Args:
            content_id (str): Unique identifier for the anime (e.g., 'ani_11111').

        Returns:
            dict: A dictionary containing a list of episodes, where each episode has number, title, air date, and runtime.

        Raises:
            KeyError: If no episodes are found for the provided content_id, suggesting the anime ID does not exist.
        """
        # Access the database instance
        db = self.db

        # Retrieve the episode table from the database
        # According to the database class, this is a dictionary of Episode objects
        episodes_table = getattr(db, 'episode', None)

        if episodes_table is None:
            # If the table itself doesn't exist or is not initialized, we raise KeyError as per schema
            raise KeyError(f"Episode database table is not available.")

        # Filter episodes belonging to the specified content_id
        anime_episodes = []
        for ep_id, ep_obj in episodes_table.items():
            if ep_obj.content_id == content_id:
                # Parse the release date to the required string format "yyyy-mm-dd"
                air_date_str = ""
                if ep_obj.release_date:
                    # Based on instructions: "yyyy-mm-dd" for date only
                    air_date_str = ep_obj.release_date.strftime("%Y-%m-%d")

                anime_episodes.append({
                    "episode_number": ep_obj.episode_number,
                    "episode_title": ep_obj.title if ep_obj.title else "Unknown Title",
                    "air_date": air_date_str,
                    "runtime_minutes": ep_obj.runtime if ep_obj.runtime else 0
                })

        # If no episodes were found for this content_id, raise KeyError as per pre-condition/raises requirement
        if not anime_episodes:
            raise KeyError(f"No episodes found for anime with ID: {content_id}")

        # Sort episodes by episode number to ensure a logical sequence
        anime_episodes.sort(key=lambda x: x["episode_number"])

        # Return the result in the format specified by the tool schema
        return {
            "episodes": anime_episodes
        }

    @is_tool()
    def filter_tv_series_by_genre(
        self, 
        genres: List[Literal["Action", "Adventure", "Animation", "Biography", "Comedy", "Crime", "Documentary", "Drama", "Family", "Fantasy", "History", "Horror", "Music", "Mystery", "Reality-TV", "Romance", "Sci-Fi", "Sport", "Thriller", "War", "Western"]], 
        match_all: bool = False, 
        limit: int = 20
    ):
        """
        Filter and retrieve TV series that belong to specified genres.

        Args:
            genres: List of genres to filter by (e.g., ["Crime", "Drama"]).
            match_all: If True, the series must contain all specified genres; 
                       if False, it must contain at least one.
            limit: Maximum number of results to return.

        Returns:
            A dictionary containing a list of matching TV series with their IDs, titles, and genre lists.
        """
        # 1. Basic validation
        if not genres:
            raise ValueError("The 'genres' parameter cannot be empty.")

        # 2. Access database tables
        tv_series_db = getattr(self.db, "tv_series", {}) or {}
        genre_db = getattr(self.db, "genre", {}) or {}
        content_genre_db = getattr(self.db, "content_genre", {}) or {}

        # 3. Build a mapping of genre_id to genre_name for easy lookup
        # Use lowercase for consistent comparison later
        genre_id_to_name = {gid: g_obj.genre_name for gid, g_obj in genre_db.items()}

        # 4. Map each TV series to its list of genre names
        # series_genre_map: { series_id: [genre_name1, genre_name2, ...] }
        series_genre_map = {}
        for cg_id, cg_obj in content_genre_db.items():
            sid = cg_obj.content_id
            gid = cg_obj.genre_id

            if sid not in series_genre_map:
                series_genre_map[sid] = []

            if gid in genre_id_to_name:
                series_genre_map[sid].append(genre_id_to_name[gid])

        # 5. Filter the TV series based on the genres provided
        results = []

        for series_id, series_obj in tv_series_db.items():
            # Get the genres associated with this specific series
            current_series_genres = series_genre_map.get(series_id, [])

            # We need to check if the input genres match the series genres.
            # Since genre names are natural language, we use fuzzy matching for robustness.
            # We'll check which of the 'input genres' are present in 'current_series_genres'.
            matched_input_genres = []
            for target_genre in genres:
                # Check if target_genre exists in current_series_genres using fuzzy matching
                # We use process.extractOne to find the best match in the series' genre list
                if current_series_genres:
                    match, score = process.extractOne(target_genre, current_series_genres, scorer=fuzz.ratio)
                    if score >= 90:  # High threshold for genre names which are usually short and distinct
                        matched_input_genres.append(target_genre)

            # Apply filtering logic (Match All vs Match Any)
            is_match = False
            if match_all:
                # Series must have matches for ALL requested genres
                if len(matched_input_genres) == len(genres):
                    is_match = True
            else:
                # Series must have a match for AT LEAST ONE requested genre
                if len(matched_input_genres) > 0:
                    is_match = True

            if is_match:
                results.append({
                    "series_id": series_obj.series_id,
                    "title": series_obj.title,
                    "genres": current_series_genres
                })

            # Stop if we've reached the limit
            if len(results) >= limit:
                break

        return {"results": results}

    @is_tool()
    def get_recently_added_movies(self, limit: int, days_back: int = None):
            """
            Retrieve a list of movies that were recently added to the database, sorted by addition date in descending order.

            Args:
                limit (int): Maximum number of recently added movies to return.
                days_back (int, optional): Number of days to look back from current date.

            Returns:
                dict: A dictionary containing the list of recently added movies.
            """
            from datetime import datetime, timedelta

            # 1. Parameter Validation
            if limit <= 0:
                raise ValueError("The 'limit' parameter must be a positive integer.")

            if days_back is not None and days_back < 0:
                raise ValueError("The 'days_back' parameter must be a non-negative integer.")

            # 2. Access the movie database
            # Use getattr to safely access the movie table in case it hasn't been initialized
            movie_table = getattr(self.db, 'movie', None)
            if not movie_table:
                return {"recently_added": []}

            # Convert dictionary values to a list for processing
            all_movies = list(movie_table.values())

            # 3. Filtering by 'days_back'
            # If days_back is provided, only include movies added within that timeframe
            if days_back is not None:
                now = datetime.now()
                threshold_date = now - timedelta(days=days_back)
                filtered_movies = [
                    movie for movie in all_movies 
                    if movie.added_date >= threshold_date
                ]
            else:
                # If days_back is not provided, we consider all movies and simply sort them
                filtered_movies = all_movies

            # 4. Sorting
            # Sort movies by added_date in descending order (most recent first)
            filtered_movies.sort(key=lambda x: x.added_date, reverse=True)

            # 5. Apply Limit
            # Take the top 'limit' movies from the sorted list
            recent_movies = filtered_movies[:limit]

            # 6. Formatting the Output
            # Transform the Movie objects into the dictionary structure required by the tool schema
            recently_added_list = []
            for movie in recent_movies:
                # added_date is a datetime object in the database, convert to "yyyy-mm-dd HH:MM:SS" string
                formatted_date = movie.added_date.strftime("%Y-%m-%d %H:%M:%S")

                recently_added_list.append({
                    "movie_id": movie.movie_id,
                    "title": movie.title,
                    "added_date": formatted_date
                })

            return {"recently_added": recently_added_list}

    @is_tool()
    def search_anime_by_voice_actor(self, actor_name: str, limit: int = 20):
        """
        Search for anime featuring a specific voice actor by voice actor name.
        """
        # Pre-condition check: Voice actor name must be a non-empty string
        if not actor_name or not isinstance(actor_name, str):
            raise ValueError("actor_name must be a non-empty string.")

        # Access the database tables
        anime_db = getattr(self.db, 'anime', {})
        person_db = getattr(self.db, 'person', {})
        cast_member_db = getattr(self.db, 'cast_member', {})

        if not person_db or not cast_member_db or not anime_db:
            return {"anime": []}

        # Step 1: Find matching persons using fuzzy matching for the actor_name
        # We extract all person names and their IDs
        person_list = list(person_db.values())
        person_names = [p.name for p in person_list]

        # Use process.extract to find matches. We use a threshold to ensure relevance.
        # Since the description says "Full or partial name", fuzzy matching is very suitable.
        matches = process.extract(actor_name, person_names, limit=None)

        # Filter matches with a reasonable score (e.g., 80)
        matched_person_ids = set()
        for match_name, score in matches:
            if score >= 80:
                # Find all person objects with this name (in case of duplicates, though unlikely for IDs)
                for p in person_list:
                    if p.name == match_name:
                        matched_person_ids.add(p.person_id)

        if not matched_person_ids:
            return {"anime": []}

        # Step 2: Search cast_member table for entries matching these person_ids and role 'voice_actor'
        results = []
        seen_combinations = set() # To avoid duplicate entries for the same anime-character-actor combo

        for cast_id, cast in cast_member_db.items():
            if cast.person_id in matched_person_ids and cast.role == "voice_actor":
                # Step 3: Get anime details
                anime = anime_db.get(cast.content_id)
                if anime:
                    # Prepare result entry
                    # anime_id, title, character_name, first_air_year

                    # Extract year from release_date (datetime object)
                    first_air_year = None
                    if anime.release_date:
                        first_air_year = anime.release_date.year

                    entry = {
                        "anime_id": anime.anime_id,
                        "title": anime.title,
                        "character_name": cast.character_name,
                        "first_air_year": first_air_year
                    }

                    # Create a unique key to prevent exact duplicates in the return list
                    unique_key = (anime.anime_id, cast.character_name)
                    if unique_key not in seen_combinations:
                        results.append(entry)
                        seen_combinations.add(unique_key)

                # Stop if we reached the limit
                if limit and len(results) >= limit:
                    break

        # Sort by year descending as a logical default for search results, or just return as found
        results.sort(key=lambda x: (x['first_air_year'] or 0), reverse=True)

        # Final limit application
        return {"anime": results[:limit] if limit else results}

    @is_tool()
    def filter_tv_series_by_content_rating(self, content_rating: Literal["TV-Y", "TV-Y7", "TV-G", "TV-PG", "TV-14", "TV-MA", "NR"], limit: int = None):
        """
        Filter and retrieve TV series by content rating classification (e.g., TV-Y, TV-G, TV-PG, TV-14, TV-MA).

        Args:
            content_rating: Content rating classification to filter by.
            limit: Maximum number of results to return.

        Returns:
            A dictionary containing a list of TV series with the specified content rating.
        """
        # Safety protection for the enum parameter and to fulfill the 'raises: ValueError' requirement in the schema
        valid_ratings = ["TV-Y", "TV-Y7", "TV-G", "TV-PG", "TV-14", "TV-MA", "NR"]
        if content_rating not in valid_ratings:
            raise ValueError(f"Invalid content_rating: {content_rating}. Must be one of {valid_ratings}")

        # Access the tv_series table from the database
        tv_series_data = getattr(self.db, "tv_series", None)

        # If the table is empty or doesn't exist, return an empty list
        if not tv_series_data:
            return {"results": []}

        results = []

        # Iterate through all TV series entries in the database
        for series in tv_series_data.values():
            # Check if the series content rating matches the requested rating
            # Exact matching is used here as content ratings are standardized categorical identifiers
            if series.content_rating == content_rating:
                results.append({
                    "series_id": series.series_id,
                    "title": series.title,
                    "content_rating": series.content_rating
                })

        # Apply limit if specified and valid
        if limit is not None and limit > 0:
            results = results[:limit]

        return {"results": results}

    @is_tool()
    def get_popular_tv_series_by_region(self, region_code: str, limit: int = None):
        # Basic input validation
        if not region_code:
            raise ValueError("region_code must be a non-empty string.")

        # Retrieve the database tables for content popularity and tv series
        # Using getattr and ensuring we handle None values by defaulting to an empty dictionary
        # This ensures stability if the database tables are not initialized
        popularity_table = getattr(self.db, 'content_popularity', None) or {}
        tv_series_table = getattr(self.db, 'tv_series', None) or {}

        if not popularity_table:
            return {"popular_series": []}

        # Filter popularity entries by the specified region_code
        # Since region_code is an ISO identifier/code, we use exact matching (==)
        # as per the instruction that identifiers/codes should use exact matching.
        regional_popularity = [
            pop_entry for pop_entry in popularity_table.values()
            if pop_entry.region == region_code
        ]

        # Sort the filtered entries by popularity_score in descending order (highest first)
        regional_popularity.sort(key=lambda x: x.popularity_score, reverse=True)

        # Apply the limit if provided
        if limit is not None:
            if limit <= 0:
                regional_popularity = []
            else:
                regional_popularity = regional_popularity[:limit]

        # Construct the final list of popular TV series
        popular_series_list = []
        for entry in regional_popularity:
            series_id = entry.content_id

            # Look up the series details in the tv_series table
            series_data = tv_series_table.get(series_id)

            # If the series exists in the tv_series database, add it to the results
            if series_data:
                popular_series_list.append({
                    "series_id": series_id,
                    "title": series_data.title,
                    "popularity_score": entry.popularity_score
                })

        return {"popular_series": popular_series_list}

    @is_tool()
    def calculate_total_runtime_for_movie_list(self, content_ids: list[str]):
        """
        Calculate the total runtime in minutes and hours for a given list of movie IDs.

        Args:
            content_ids (list[str]): List of movie identifiers to calculate total runtime for.

        Returns:
            dict: A dictionary containing total_runtime_minutes (int) and total_runtime_hours (float).

        Raises:
            KeyError: If any of the provided movie IDs are not found in the database.
        """
        # Access the database and the movie table
        db = self.db
        movie_table = getattr(db, "movie", {})

        if not movie_table:
            # If the table is missing or empty and we have IDs to look up, 
            # it's appropriate to raise a KeyError as the pre-condition isn't met.
            if content_ids:
                raise KeyError("Movie database is empty or not initialized.")
            return {"total_runtime_minutes": 0, "total_runtime_hours": 0.0}

        total_minutes = 0

        # Iterate through each provided movie ID
        for movie_id in content_ids:
            # Check if the movie exists in the database
            if movie_id not in movie_table:
                raise KeyError(f"Movie ID '{movie_id}' not found in the database.")

            movie_obj = movie_table[movie_id]

            # Add the runtime to the total if it exists
            # According to schema, runtime is Optional[int]
            if movie_obj.runtime is not None:
                total_minutes += movie_obj.runtime
            else:
                # If runtime is not specified, we treat it as 0
                total_minutes += 0

        # Calculate the total runtime in hours
        total_hours = float(total_minutes) / 60.0

        return {
            "total_runtime_minutes": total_minutes,
            "total_runtime_hours": round(total_hours, 2) if total_hours > 0 else 0.0
        }

    @is_tool()
    def get_related_tv_series(self, content_id: str, limit: int = 10):
        # Access the database tables safely
        tv_series_table = getattr(self.db, "tv_series", {}) or {}
        content_genre_table = getattr(self.db, "content_genre", {}) or {}
        cast_member_table = getattr(self.db, "cast_member", {}) or {}
        crew_member_table = getattr(self.db, "crew_member", {}) or {}

        # 1. Validate existence of the reference TV series
        # The content_id provided refers to the series_id in the TvSeries table
        if content_id not in tv_series_table:
            raise KeyError(f"TV series with ID '{content_id}' not found in the database.")

        source_series = tv_series_table[content_id]

        # 2. Extract comparison features for the source TV series
        # Get the set of genre IDs linked to this series
        source_genres = {
            cg.genre_id for cg in content_genre_table.values() 
            if cg.content_id == content_id
        }

        # Get the set of person IDs involved in the cast
        source_cast = {
            cm.person_id for cm in cast_member_table.values() 
            if cm.content_id == content_id
        }

        # Get the set of person IDs involved in the crew
        source_crew = {
            crm.person_id for crm in crew_member_table.values() 
            if crm.content_id == content_id
        }

        # Get plot synopsis for theme/topic comparison
        source_synopsis = source_series.plot_synopsis or ""

        # 3. Iterate through all other TV series to calculate similarity scores
        scored_series = []
        for other_id, other_series in tv_series_table.items():
            # Do not compare the series with itself
            if other_id == content_id:
                continue

            # Extract features for the candidate series
            other_genres = {
                cg.genre_id for cg in content_genre_table.values() 
                if cg.content_id == other_id
            }
            other_cast = {
                cm.person_id for cm in cast_member_table.values() 
                if cm.content_id == other_id
            }
            other_crew = {
                crm.person_id for crm in crew_member_table.values() 
                if crm.content_id == other_id
            }
            other_synopsis = other_series.plot_synopsis or ""

            # Calculate similarity components

            # Genre similarity: Jaccard index (intersection over union)
            genre_sim = 0.0
            if source_genres or other_genres:
                union_genres = source_genres | other_genres
                if union_genres:
                    genre_sim = len(source_genres & other_genres) / len(union_genres)

            # Cast similarity: Jaccard index
            cast_sim = 0.0
            if source_cast or other_cast:
                union_cast = source_cast | other_cast
                if union_cast:
                    cast_sim = len(source_cast & other_cast) / len(union_cast)

            # Crew similarity: Jaccard index
            crew_sim = 0.0
            if source_crew or other_crew:
                union_crew = source_crew | other_crew
                if union_crew:
                    crew_sim = len(source_crew & other_crew) / len(union_crew)

            # Theme similarity: Using fuzzy matching on the plot synopsis
            # token_set_ratio is robust against different word orders and partial matches
            theme_sim = 0.0
            if source_synopsis and other_synopsis:
                theme_sim = fuzz.token_set_ratio(source_synopsis, other_synopsis) / 100.0

            # Combine weighted scores to get a final similarity score
            # Distribution: Genre (40%), Themes (30%), Cast (15%), Crew (15%)
            final_score = (genre_sim * 0.4) + (theme_sim * 0.3) + (cast_sim * 0.15) + (crew_sim * 0.15)

            # Only include series with some level of similarity
            if final_score > 0:
                scored_series.append({
                    "series_id": other_id,
                    "title": other_series.title,
                    "similarity_score": round(final_score, 2)
                })

        # 4. Rank by similarity score descending
        scored_series.sort(key=lambda x: x["similarity_score"], reverse=True)

        # 5. Return the top results based on the limit
        return {
            "related_series": scored_series[:limit]
        }
