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 pet_care."""

class PetCareTools(ToolKitBase):
    """All tools for pet_care."""
    
    db: PetCareDB
    
    def __init__(self, db: PetCareDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def cancel_appointment(self, appointment_id: str, cancellation_reason: str, cancellation_timestamp: str = None):
        """
        Cancel a scheduled appointment

        Args:
            appointment_id: Unique identifier of the appointment to cancel
            cancellation_reason: Reason for cancellation
            cancellation_timestamp: Timestamp of cancellation in yyyy-mm-dd HH:MM:SS format (optional, defaults to current time)

        Returns:
            dict: Contains success status, refund eligibility, and cancellation timestamp

        Raises:
            KeyError: If appointment_id does not exist in the database
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the appointment table
        appointment_table = getattr(db, "appointment", None)

        # Check if appointment table exists
        if appointment_table is None:
            raise KeyError(f"Appointment table not found in database")

        # Check if the appointment exists
        if appointment_id not in appointment_table:
            raise KeyError(f"Appointment with ID '{appointment_id}' does not exist")

        # Retrieve the appointment
        appointment = appointment_table[appointment_id]

        # Check if appointment is already cancelled
        if appointment.status == "cancelled":
            raise ValueError(f"Appointment '{appointment_id}' is already cancelled")

        # Determine cancellation timestamp
        if cancellation_timestamp is None:
            cancellation_dt = datetime.now()
        else:
            # Parse the cancellation timestamp string to datetime object
            try:
                cancellation_dt = datetime.strptime(cancellation_timestamp, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                # Try parsing date-only format
                try:
                    cancellation_dt = datetime.strptime(cancellation_timestamp, "%Y-%m-%d")
                except ValueError:
                    raise ValueError(f"Invalid timestamp format. Expected 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd', got '{cancellation_timestamp}'")

        # Calculate refund eligibility based on cancellation time
        # Refund is eligible if cancelled at least 24 hours before the appointment
        time_until_appointment = (appointment.appointment_datetime - cancellation_dt).total_seconds()
        hours_until_appointment = time_until_appointment / 3600
        refund_eligible = hours_until_appointment >= 24

        # Update the appointment with cancellation details
        appointment.status = "cancelled"
        appointment.cancellation_reason = cancellation_reason
        appointment.cancellation_timestamp = cancellation_dt

        # Save the updated appointment back to the database
        appointment_table[appointment_id] = appointment
        setattr(db, "appointment", appointment_table)

        # Format the cancellation timestamp for return
        cancellation_timestamp_str = cancellation_dt.strftime("%Y-%m-%d %H:%M:%S")

        # Return success response
        return {
            "success": True,
            "refund_eligible": refund_eligible,
            "cancellation_timestamp": cancellation_timestamp_str
        }

    @is_tool()
    def reschedule_appointment(self, appointment_id: str, new_appointment_datetime: str, reason: str = None):
        """
        Reschedule an existing appointment to a new date and time

        Args:
            appointment_id: Unique identifier of the appointment to reschedule
            new_appointment_datetime: New date and time for the appointment in yyyy-mm-dd HH:MM:SS format
            reason: Reason for rescheduling (optional)

        Returns:
            Dictionary containing:
            - success: Boolean indicating whether the rescheduling was successful
            - old_datetime: Previous appointment date and time in yyyy-mm-dd HH:MM:SS format
            - new_datetime: New appointment date and time in yyyy-mm-dd HH:MM:SS format
            - updated_timestamp: Timestamp when the rescheduling was completed in yyyy-mm-dd HH:MM:SS format

        Raises:
            KeyError: If appointment_id does not exist
            ValueError: If new_appointment_datetime format is invalid
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the appointment table
        appointment_table = getattr(db, "appointment", None)

        # Check if appointment table exists and is not empty
        if appointment_table is None or not isinstance(appointment_table, dict):
            raise KeyError(f"Appointment with ID '{appointment_id}' not found")

        # Check if the appointment exists
        if appointment_id not in appointment_table:
            raise KeyError(f"Appointment with ID '{appointment_id}' not found")

        # Parse the new appointment datetime string
        try:
            new_datetime_obj = datetime.strptime(new_appointment_datetime, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid datetime format for new_appointment_datetime. Expected 'yyyy-mm-dd HH:MM:SS', got '{new_appointment_datetime}'")

        # Get the existing appointment
        appointment = appointment_table[appointment_id]

        # Store the old datetime for return value
        old_datetime_obj = appointment.appointment_datetime
        old_datetime_str = old_datetime_obj.strftime("%Y-%m-%d %H:%M:%S")

        # Update the appointment with new datetime
        appointment.appointment_datetime = new_datetime_obj

        # If reason is provided, add it to special_requests or update it
        if reason:
            if appointment.special_requests:
                appointment.special_requests += f" | Rescheduled: {reason}"
            else:
                appointment.special_requests = f"Rescheduled: {reason}"

        # Generate the current timestamp for when the rescheduling was completed
        updated_timestamp_obj = datetime.now()
        updated_timestamp_str = updated_timestamp_obj.strftime("%Y-%m-%d %H:%M:%S")

        # Update the appointment in the database
        appointment_table[appointment_id] = appointment
        setattr(db, "appointment", appointment_table)

        # Return the result
        return {
            "success": True,
            "old_datetime": old_datetime_str,
            "new_datetime": new_appointment_datetime,
            "updated_timestamp": updated_timestamp_str
        }

    @is_tool()
    def get_active_reminders(self, pet_id: str, include_completed: bool = False):
        # Get database instance
        db = self.db

        # Retrieve wellness_reminder table from database
        wellness_reminder_table = getattr(db, 'wellness_reminder', None)

        # Check if wellness_reminder table exists
        if wellness_reminder_table is None:
            raise KeyError("wellness_reminder table not found in database")

        # Check if pet exists by verifying if there are any reminders for this pet_id
        pet_reminders_exist = any(
            reminder.pet_id == pet_id 
            for reminder in wellness_reminder_table.values()
        )

        # If no reminders exist for this pet, raise KeyError (pet profile must exist)
        if not pet_reminders_exist:
            raise KeyError(f"No reminders found for pet_id: {pet_id}. Pet profile may not exist in the system")

        # Get current date for comparison
        current_date = datetime.now().date()

        # Filter reminders based on pet_id and status
        filtered_reminders = []

        for reminder in wellness_reminder_table.values():
            # Check if reminder belongs to the specified pet
            if reminder.pet_id != pet_id:
                continue

            # If include_completed is False, only include active reminders
            # If include_completed is True, include both active and completed reminders
            if include_completed:
                # Include all reminders regardless of status
                filtered_reminders.append(reminder)
            else:
                # Only include reminders with status "active"
                if reminder.status == "active":
                    filtered_reminders.append(reminder)

        # Convert reminder objects to dictionary format for output
        reminder_list = []
        urgent_count = 0

        for reminder in filtered_reminders:
            # Create reminder dictionary with all relevant fields
            reminder_dict = {
                "reminder_id": reminder.reminder_id,
                "reminder_type": reminder.reminder_type,
                "reminder_date": reminder.reminder_date.strftime("%Y-%m-%d"),
                "title": reminder.title,
                "description": reminder.description,
                "priority": reminder.priority,
                "status": reminder.status
            }

            # Add optional fields if they exist
            if reminder.advance_notice_days is not None:
                reminder_dict["advance_notice_days"] = reminder.advance_notice_days

            if reminder.completion_date is not None:
                reminder_dict["completion_date"] = reminder.completion_date.strftime("%Y-%m-%d")

            if reminder.completion_notes is not None:
                reminder_dict["completion_notes"] = reminder.completion_notes

            reminder_list.append(reminder_dict)

            # Count urgent reminders
            # A reminder is considered urgent if:
            # 1. Priority is explicitly set to "urgent", OR
            # 2. Reminder date is within advance_notice_days or has passed
            if reminder.priority == "urgent":
                urgent_count += 1
            elif reminder.advance_notice_days is not None:
                # Calculate if reminder is within advance notice period
                days_until_reminder = (reminder.reminder_date - current_date).days
                if days_until_reminder <= reminder.advance_notice_days:
                    urgent_count += 1
            elif reminder.reminder_date <= current_date:
                # Reminder date has passed or is today
                urgent_count += 1

        # Return results in the required format
        return {
            "reminders": reminder_list,
            "total_count": len(reminder_list),
            "urgent_count": urgent_count
        }

    @is_tool()
    def get_health_checkup_history(self, pet_id: str, start_date: str = None, end_date: str = None):
        # Get database instance
        db = self.db

        # Retrieve health_checkup_record table from database
        health_checkup_table = getattr(db, "health_checkup_record", None)

        # Validate that the health_checkup_record table exists
        if health_checkup_table is None:
            raise KeyError("health_checkup_record table not found in database")

        # Initialize result list to store matching checkup records
        matching_checkups = []

        # Parse date range if provided
        # Convert string dates to date objects for comparison
        start_date_obj = None
        end_date_obj = None

        if start_date:
            try:
                start_date_obj = datetime.strptime(start_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid start_date format: {start_date}. Expected format: yyyy-mm-dd")

        if end_date:
            try:
                end_date_obj = datetime.strptime(end_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid end_date format: {end_date}. Expected format: yyyy-mm-dd")

        # Validate date range logic
        if start_date_obj and end_date_obj and start_date_obj > end_date_obj:
            raise ValueError("start_date cannot be later than end_date")

        # Iterate through all health checkup records
        for checkup_id, checkup_record in health_checkup_table.items():
            # Filter by pet_id - must match exactly
            if checkup_record.pet_id != pet_id:
                continue

            # Filter by date range if specified
            checkup_date = checkup_record.checkup_date

            # Check if checkup_date falls within the specified range
            if start_date_obj and checkup_date < start_date_obj:
                continue

            if end_date_obj and checkup_date > end_date_obj:
                continue

            # Build checkup record dictionary with all relevant information
            checkup_dict = {
                "checkup_id": checkup_record.checkup_id,
                "checkup_date": checkup_record.checkup_date.strftime("%Y-%m-%d"),
                "checkup_type": checkup_record.checkup_type,
                "weight_kg": checkup_record.weight_kg,
                "temperature_celsius": checkup_record.temperature_celsius,
                "heart_rate_bpm": checkup_record.heart_rate_bpm,
                "respiratory_rate": checkup_record.respiratory_rate,
                "veterinarian_name": checkup_record.veterinarian_name,
                "clinic_id": checkup_record.clinic_id,
                "findings": checkup_record.findings,
                "recommendations": checkup_record.recommendations,
                "creation_timestamp": checkup_record.creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")
            }

            matching_checkups.append(checkup_dict)

        # Sort checkups by checkup_date in descending order (most recent first)
        matching_checkups.sort(key=lambda x: x["checkup_date"], reverse=True)

        # Return results with total count
        return {
            "checkups": matching_checkups,
            "total_count": len(matching_checkups)
        }

    @is_tool()
    def complete_reminder(self, reminder_id: str, completion_date: str, completion_notes: str = None):
        """
        Mark a reminder as completed

        Args:
            reminder_id: Unique identifier of the reminder to complete
            completion_date: Date when the reminder was completed in yyyy-mm-dd format
            completion_notes: Optional notes about the completion

        Returns:
            dict: Contains success status and completion timestamp

        Raises:
            KeyError: If the reminder does not exist
            ValueError: If completion_date format is invalid or reminder is not active
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the wellness_reminder table
        wellness_reminder_table = getattr(db, "wellness_reminder", None)

        # Check if the table exists
        if wellness_reminder_table is None:
            raise KeyError(f"Reminder with ID '{reminder_id}' does not exist: wellness_reminder table not found")

        # Check if the reminder exists
        if reminder_id not in wellness_reminder_table:
            raise KeyError(f"Reminder with ID '{reminder_id}' does not exist")

        # Get the reminder
        reminder = wellness_reminder_table[reminder_id]

        # Verify the reminder is active (pre-condition check)
        if reminder.status != "active":
            raise ValueError(f"Reminder with ID '{reminder_id}' is not active (current status: {reminder.status})")

        # Parse and validate completion_date format
        try:
            completion_date_obj = datetime.strptime(completion_date, "%Y-%m-%d").date()
        except ValueError as e:
            raise ValueError(f"Invalid completion_date format. Expected 'yyyy-mm-dd', got '{completion_date}': {str(e)}")

        # Update the reminder with completion information
        reminder.status = "completed"
        reminder.completion_date = completion_date_obj
        if completion_notes is not None:
            reminder.completion_notes = completion_notes

        # Generate completion timestamp in the required format
        completion_timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Update the reminder in the database
        wellness_reminder_table[reminder_id] = reminder
        setattr(db, "wellness_reminder", wellness_reminder_table)

        # Return success response with completion timestamp
        return {
            "success": True,
            "completion_timestamp": completion_timestamp
        }

    @is_tool()
    def update_pet_weight(self, pet_id: str, weight_kg: float, measurement_date: str):
        """
        Update the current weight of a pet and add a new weight history record.

        Args:
            pet_id: Unique identifier of the pet
            weight_kg: New weight of the pet in kilograms
            measurement_date: Date when the weight was measured in yyyy-mm-dd format

        Returns:
            dict: Contains success status and updated timestamp

        Raises:
            KeyError: If pet profile does not exist
            ValueError: If weight_kg is negative or measurement_date format is invalid
        """
        from datetime import datetime, date
        import secrets
        import hashlib

        # Access database instance
        db = self.db

        # Validate weight_kg parameter
        if weight_kg < 0:
            raise ValueError("weight_kg must be a non-negative number")

        # Parse and validate measurement_date format
        try:
            parsed_measurement_date = datetime.strptime(measurement_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError("measurement_date must be in yyyy-mm-dd format")

        # Get pet_profile table from database
        pet_profile_table = getattr(db, "pet_profile", None)
        if pet_profile_table is None:
            raise KeyError("pet_profile table does not exist in database")

        # Check if pet exists in the database
        if pet_id not in pet_profile_table:
            raise KeyError(f"Pet with pet_id '{pet_id}' does not exist")

        # Get the pet profile
        pet = pet_profile_table[pet_id]

        # Update the pet's current weight
        pet.weight_kg = weight_kg

        # Update pet_profile table in database
        setattr(db, "pet_profile", pet_profile_table)

        # Generate current timestamp for the update
        current_timestamp = datetime.now()

        # Create a new weight history record
        # Generate unique weight_history_id
        weight_history_id = "wh_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create new WeightHistory instance
        new_weight_history = WeightHistory(
            weight_history_id=weight_history_id,
            pet_id=pet_id,
            weight_kg=weight_kg,
            measurement_date=parsed_measurement_date,
            updated_timestamp=current_timestamp
        )

        # Get weight_history table from database
        weight_history_table = getattr(db, "weight_history", None)
        if weight_history_table is None:
            # Initialize weight_history table if it doesn't exist
            weight_history_table = {}

        # Add new weight history record to the table
        weight_history_table[weight_history_id] = new_weight_history

        # Update weight_history table in database
        setattr(db, "weight_history", weight_history_table)

        # Format updated_timestamp as string in yyyy-mm-dd HH:MM:SS format
        updated_timestamp_str = current_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return success response
        return {
            "success": True,
            "updated_timestamp": updated_timestamp_str
        }

    @is_tool()
    def create_health_checkup_record(
        self,
        pet_id: str,
        checkup_date: str,
        checkup_type: Literal["routine", "emergency", "follow_up", "pre_surgery", "post_surgery", "dental", "specialized"],
        veterinarian_name: str,
        weight_kg: float = None,
        temperature_celsius: float = None,
        heart_rate_bpm: int = None,
        respiratory_rate: int = None,
        clinic_id: str = None,
        findings: str = None,
        recommendations: str = None
    ) -> dict:
        """
        Create a new health checkup record for a pet.

        This method validates input parameters, generates a unique checkup ID,
        creates a HealthCheckupRecord instance, and stores it in the database.

        Args:
            pet_id: Unique identifier of the pet
            checkup_date: Date when checkup was performed (format: yyyy-mm-dd)
            checkup_type: Type of health checkup (must be one of the enum values)
            veterinarian_name: Name of the veterinarian who performed the checkup
            weight_kg: Optional weight measured during checkup in kilograms
            temperature_celsius: Optional body temperature in Celsius
            heart_rate_bpm: Optional heart rate in beats per minute
            respiratory_rate: Optional respiratory rate per minute
            clinic_id: Optional unique identifier of the clinic
            findings: Optional general findings and observations
            recommendations: Optional veterinarian recommendations

        Returns:
            dict: Contains checkup_id and creation_timestamp

        Raises:
            ValueError: If required parameters are missing or invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters
        if not pet_id or not isinstance(pet_id, str) or not pet_id.strip():
            raise ValueError("pet_id is required and must be a non-empty string")

        if not checkup_date or not isinstance(checkup_date, str):
            raise ValueError("checkup_date is required and must be a string")

        if not checkup_type or not isinstance(checkup_type, str):
            raise ValueError("checkup_type is required and must be a string")

        if not veterinarian_name or not isinstance(veterinarian_name, str) or not veterinarian_name.strip():
            raise ValueError("veterinarian_name is required and must be a non-empty string")

        # Validate checkup_type enum value (with safety protection)
        valid_checkup_types = ["routine", "emergency", "follow_up", "pre_surgery", "post_surgery", "dental", "specialized"]
        if checkup_type not in valid_checkup_types:
            raise ValueError(f"checkup_type must be one of {valid_checkup_types}, got '{checkup_type}'")

        # Parse and validate checkup_date format
        try:
            parsed_date = datetime.strptime(checkup_date, "%Y-%m-%d").date()
        except ValueError as e:
            raise ValueError(f"checkup_date must be in yyyy-mm-dd format, got '{checkup_date}': {str(e)}")

        # Validate optional numeric parameters
        if weight_kg is not None:
            if not isinstance(weight_kg, (int, float)) or weight_kg <= 0:
                raise ValueError("weight_kg must be a positive number")

        if temperature_celsius is not None:
            if not isinstance(temperature_celsius, (int, float)):
                raise ValueError("temperature_celsius must be a number")

        if heart_rate_bpm is not None:
            if not isinstance(heart_rate_bpm, int) or heart_rate_bpm <= 0:
                raise ValueError("heart_rate_bpm must be a positive integer")

        if respiratory_rate is not None:
            if not isinstance(respiratory_rate, int) or respiratory_rate <= 0:
                raise ValueError("respiratory_rate must be a positive integer")

        # Generate unique checkup_id
        checkup_id = "checkup_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for record creation
        creation_timestamp = datetime.now()

        # Access database
        db = self.db

        # Initialize health_checkup_record table if it doesn't exist
        if getattr(db, "health_checkup_record", None) is None:
            setattr(db, "health_checkup_record", {})

        # Get the health_checkup_record table
        health_checkup_table = getattr(db, "health_checkup_record")

        # Create new HealthCheckupRecord instance
        new_checkup_record = HealthCheckupRecord(
            checkup_id=checkup_id,
            pet_id=pet_id,
            checkup_date=parsed_date,
            checkup_type=checkup_type,
            weight_kg=weight_kg,
            temperature_celsius=temperature_celsius,
            heart_rate_bpm=heart_rate_bpm,
            respiratory_rate=respiratory_rate,
            veterinarian_name=veterinarian_name,
            clinic_id=clinic_id,
            findings=findings,
            recommendations=recommendations,
            creation_timestamp=creation_timestamp
        )

        # Store the record in the database
        health_checkup_table[checkup_id] = new_checkup_record
        setattr(db, "health_checkup_record", health_checkup_table)

        # Format creation_timestamp as string for return value
        creation_timestamp_str = creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "checkup_id": checkup_id,
            "creation_timestamp": creation_timestamp_str
        }

    @is_tool()
    def track_training_progress(
        self,
        pet_id: str,
        command_name: str,
        start_date: Optional[str] = None,
        end_date: Optional[str] = None
    ) -> dict:
        """
        Track training progress for specific commands or skills over time.

        This method analyzes training session records for a specific pet and command,
        calculating metrics like total sessions, average success rate, progress trend,
        and mastery level. It optionally filters sessions within a date range.

        Args:
            pet_id: Unique identifier of the pet
            command_name: Specific command to track progress for
            start_date: Optional start date in yyyy-mm-dd format
            end_date: Optional end date in yyyy-mm-dd format

        Returns:
            Dictionary containing training progress analysis with keys:
            - total_sessions: Total number of training sessions
            - average_success_rate: Average success rate across sessions
            - progress_trend: Trend in learning ("improving", "stable", "declining")
            - mastery_level: Current mastery level ("beginner", "intermediate", "advanced", "mastered")
            - recommendations: Training recommendations based on progress

        Raises:
            KeyError: If pet has no training session records or required data is missing
        """
        from datetime import datetime
        from thefuzz import fuzz

        # Access the database
        db = self.db

        # Get training session table
        training_sessions = getattr(db, "training_session", None)
        if training_sessions is None or len(training_sessions) == 0:
            raise KeyError(f"No training session records found in database")

        # Parse date filters if provided
        start_datetime = None
        end_datetime = None
        if start_date:
            try:
                start_datetime = datetime.strptime(start_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid start_date format: {start_date}. Expected yyyy-mm-dd")

        if end_date:
            try:
                end_datetime = datetime.strptime(end_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid end_date format: {end_date}. Expected yyyy-mm-dd")

        # Filter sessions for the specific pet and command
        # Use fuzzy matching for command name to improve robustness
        matching_sessions = []
        for session_id, session in training_sessions.items():
            # Check if session belongs to the specified pet
            if session.pet_id != pet_id:
                continue

            # Check if session is within date range (if specified)
            if start_datetime and session.session_date < start_datetime:
                continue
            if end_datetime and session.session_date > end_datetime:
                continue

            # Check if the command was practiced in this session
            # commands_practiced is a string that may contain multiple commands
            if session.commands_practiced:
                # Use fuzzy matching to find if command_name is mentioned in commands_practiced
                # Split commands_practiced by common delimiters (comma, semicolon, etc.)
                practiced_commands = [
                    cmd.strip() 
                    for cmd in session.commands_practiced.replace(';', ',').split(',')
                ]

                # Check if any practiced command matches the target command (fuzzy match)
                command_found = False
                for practiced_cmd in practiced_commands:
                    if fuzz.partial_ratio(command_name.lower(), practiced_cmd.lower()) >= 80:
                        command_found = True
                        break

                if command_found:
                    matching_sessions.append(session)

        # Check if any matching sessions were found
        if len(matching_sessions) == 0:
            raise KeyError(f"No training sessions found for pet_id '{pet_id}' with command '{command_name}'")

        # Sort sessions by date to analyze progress over time
        matching_sessions.sort(key=lambda s: s.session_date)

        # Calculate total sessions
        total_sessions = len(matching_sessions)

        # Calculate average success rate
        # Only include sessions that have a success_rate value
        sessions_with_success_rate = [s for s in matching_sessions if s.success_rate is not None]
        if len(sessions_with_success_rate) == 0:
            # If no sessions have success rate, use 0 as default
            average_success_rate = 0.0
        else:
            average_success_rate = sum(s.success_rate for s in sessions_with_success_rate) / len(sessions_with_success_rate)

        # Determine progress trend by comparing first half vs second half of sessions
        progress_trend = "stable"
        if len(sessions_with_success_rate) >= 4:
            # Split sessions into two halves
            mid_point = len(sessions_with_success_rate) // 2
            first_half = sessions_with_success_rate[:mid_point]
            second_half = sessions_with_success_rate[mid_point:]

            # Calculate average success rates for each half
            first_half_avg = sum(s.success_rate for s in first_half) / len(first_half)
            second_half_avg = sum(s.success_rate for s in second_half) / len(second_half)

            # Determine trend based on improvement threshold (10% difference)
            improvement = second_half_avg - first_half_avg
            if improvement > 10:
                progress_trend = "improving"
            elif improvement < -10:
                progress_trend = "declining"
            else:
                progress_trend = "stable"
        elif len(sessions_with_success_rate) >= 2:
            # For fewer sessions, compare first and last
            first_rate = sessions_with_success_rate[0].success_rate
            last_rate = sessions_with_success_rate[-1].success_rate
            improvement = last_rate - first_rate

            if improvement > 10:
                progress_trend = "improving"
            elif improvement < -10:
                progress_trend = "declining"
            else:
                progress_trend = "stable"

        # Determine mastery level based on average success rate and number of sessions
        mastery_level = "beginner"
        if average_success_rate >= 90 and total_sessions >= 8:
            mastery_level = "mastered"
        elif average_success_rate >= 75 and total_sessions >= 5:
            mastery_level = "advanced"
        elif average_success_rate >= 60 and total_sessions >= 3:
            mastery_level = "intermediate"
        else:
            mastery_level = "beginner"

        # Generate recommendations based on progress analysis
        recommendations = ""
        if mastery_level == "mastered":
            recommendations = "Excellent progress! Consider introducing more complex variations or combining this command with others."
        elif mastery_level == "advanced":
            recommendations = "Great work! Continue practicing in different environments to reinforce learning and work towards mastery."
        elif mastery_level == "intermediate":
            recommendations = "Good progress. Increase practice frequency and consistency to advance to the next level."
        else:  # beginner
            if progress_trend == "improving":
                recommendations = "Keep up the good work! Continue regular practice sessions to build consistency."
            elif progress_trend == "declining":
                recommendations = "Consider reviewing training techniques and ensuring consistency in commands and rewards. May need to simplify the approach."
            else:  # stable
                recommendations = "Continue regular practice sessions. Try using higher-value rewards and shorter, more frequent training sessions."

        # Additional recommendations based on trend
        if progress_trend == "declining" and mastery_level != "beginner":
            recommendations += " Monitor for any environmental changes or health issues that might be affecting performance."
        elif progress_trend == "stable" and total_sessions >= 10:
            recommendations += " Consider consulting with a professional trainer to identify potential training adjustments."

        return {
            "total_sessions": total_sessions,
            "average_success_rate": round(average_success_rate, 1),
            "progress_trend": progress_trend,
            "mastery_level": mastery_level,
            "recommendations": recommendations
        }

    @is_tool()
    def create_grooming_record(
        self,
        pet_id: str,
        grooming_date: str,
        services_performed: List[Literal["bath", "haircut", "nail_trim", "ear_cleaning", "teeth_brushing", "anal_gland_expression", "de_shedding", "flea_treatment"]],
        groomer_name: str,
        location_id: Optional[str] = None,
        duration_minutes: Optional[int] = None,
        coat_condition: Optional[Literal["excellent", "good", "fair", "poor"]] = None,
        notes: Optional[str] = None
    ) -> dict:
        """
        Create a grooming service record for a pet.
        """
        from datetime import datetime
        import secrets
        import hashlib
        import json

        # Validate required parameters
        if not pet_id or not isinstance(pet_id, str):
            raise ValueError("pet_id must be a non-empty string")

        if not grooming_date or not isinstance(grooming_date, str):
            raise ValueError("grooming_date must be a non-empty string")

        if not services_performed or not isinstance(services_performed, list) or len(services_performed) == 0:
            raise ValueError("services_performed must be a non-empty list")

        if not groomer_name or not isinstance(groomer_name, str):
            raise ValueError("groomer_name must be a non-empty string")

        # Validate and parse grooming_date format (yyyy-mm-dd)
        try:
            parsed_date = datetime.strptime(grooming_date, "%Y-%m-%d").date()
        except ValueError as e:
            raise ValueError(f"grooming_date must be in yyyy-mm-dd format: {str(e)}")

        # Validate services_performed enum values
        valid_services = ["bath", "haircut", "nail_trim", "ear_cleaning", "teeth_brushing", 
                         "anal_gland_expression", "de_shedding", "flea_treatment"]
        for service in services_performed:
            if service not in valid_services:
                raise ValueError(f"Invalid service '{service}'. Must be one of: {', '.join(valid_services)}")

        # Validate optional duration_minutes
        if duration_minutes is not None:
            if not isinstance(duration_minutes, int) or duration_minutes <= 0:
                raise ValueError("duration_minutes must be a positive integer")

        # Validate optional coat_condition enum value
        if coat_condition is not None:
            valid_conditions = ["excellent", "good", "fair", "poor"]
            if coat_condition not in valid_conditions:
                raise ValueError(f"Invalid coat_condition '{coat_condition}'. Must be one of: {', '.join(valid_conditions)}")

        # Generate unique grooming record ID using hash-based approach
        prefix = "groom_"
        grooming_record_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create creation timestamp
        creation_timestamp = datetime.now()

        # Convert services_performed list to JSON string for storage
        services_str = json.dumps(services_performed)

        # Access database
        db = self.db

        # Initialize grooming_record table if it doesn't exist
        if getattr(db, "grooming_record", None) is None:
            setattr(db, "grooming_record", {})

        # Get grooming_record table
        grooming_record_table = getattr(db, "grooming_record")

        # Import GroomingRecord class from database module (correct absolute import)

        # Create new GroomingRecord instance
        new_record = GroomingRecord(
            grooming_record_id=grooming_record_id,
            pet_id=pet_id,
            grooming_date=parsed_date,
            services_performed=services_str,
            groomer_name=groomer_name,
            location_id=location_id,
            duration_minutes=duration_minutes,
            coat_condition=coat_condition,
            notes=notes,
            creation_timestamp=creation_timestamp
        )

        # Store the record in the database
        grooming_record_table[grooming_record_id] = new_record
        setattr(db, "grooming_record", grooming_record_table)

        # Format creation_timestamp for return (yyyy-mm-dd HH:MM:SS)
        formatted_timestamp = creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result with grooming_record_id and formatted timestamp
        return {
            "grooming_record_id": grooming_record_id,
            "creation_timestamp": formatted_timestamp
        }

    @is_tool()
    def create_medication_record(
        self,
        pet_id: str,
        medication_name: str,
        dosage: str,
        frequency: str,
        start_date: str,
        end_date: str = None,
        purpose: str = None,
        veterinarian_name: str = None,
        special_instructions: str = None
    ):
        """
        Create a new medication record for a pet.

        This method validates the input parameters, generates a unique medication ID,
        and creates a new medication record in the database.

        Args:
            pet_id: Unique identifier of the pet
            medication_name: Name of the medication
            dosage: Dosage amount (e.g., "250mg")
            frequency: Frequency of administration (e.g., "twice daily")
            start_date: Date when medication started in yyyy-mm-dd format
            end_date: Optional date when medication should end in yyyy-mm-dd format
            purpose: Optional purpose or condition being treated
            veterinarian_name: Optional name of the veterinarian who prescribed
            special_instructions: Optional special instructions for administration

        Returns:
            dict: Contains medication_id and creation_timestamp

        Raises:
            ValueError: If required parameters are invalid or dates are in wrong format
        """
        from datetime import datetime, date
        import secrets
        import hashlib

        # Validate required parameters are not empty
        if not pet_id or not isinstance(pet_id, str) or not pet_id.strip():
            raise ValueError("pet_id must be a non-empty string")

        if not medication_name or not isinstance(medication_name, str) or not medication_name.strip():
            raise ValueError("medication_name must be a non-empty string")

        if not dosage or not isinstance(dosage, str) or not dosage.strip():
            raise ValueError("dosage must be a non-empty string")

        if not frequency or not isinstance(frequency, str) or not frequency.strip():
            raise ValueError("frequency must be a non-empty string")

        if not start_date or not isinstance(start_date, str) or not start_date.strip():
            raise ValueError("start_date must be a non-empty string")

        # Parse and validate start_date format
        try:
            parsed_start_date = datetime.strptime(start_date, "%Y-%m-%d").date()
        except ValueError as e:
            raise ValueError(f"start_date must be in yyyy-mm-dd format: {str(e)}")

        # Parse and validate end_date format if provided
        parsed_end_date = None
        if end_date is not None:
            if not isinstance(end_date, str) or not end_date.strip():
                raise ValueError("end_date must be a non-empty string if provided")
            try:
                parsed_end_date = datetime.strptime(end_date, "%Y-%m-%d").date()
            except ValueError as e:
                raise ValueError(f"end_date must be in yyyy-mm-dd format: {str(e)}")

            # Validate that end_date is not before start_date
            if parsed_end_date < parsed_start_date:
                raise ValueError("end_date cannot be before start_date")

        # Generate unique medication ID using secure random hash
        medication_id = "med_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for creation_timestamp
        creation_timestamp = datetime.now()

        # Access the database
        db = self.db

        # Initialize medication_record table if it doesn't exist
        if getattr(db, "medication_record", None) is None:
            setattr(db, "medication_record", {})

        # Create new MedicationRecord instance
        new_medication_record = MedicationRecord(
            medication_id=medication_id,
            pet_id=pet_id.strip(),
            medication_name=medication_name.strip(),
            dosage=dosage.strip(),
            frequency=frequency.strip(),
            start_date=parsed_start_date,
            end_date=parsed_end_date,
            purpose=purpose.strip() if purpose and isinstance(purpose, str) else purpose,
            veterinarian_name=veterinarian_name.strip() if veterinarian_name and isinstance(veterinarian_name, str) else veterinarian_name,
            special_instructions=special_instructions.strip() if special_instructions and isinstance(special_instructions, str) else special_instructions,
            creation_timestamp=creation_timestamp
        )

        # Get current medication_record table data
        medication_record_table = getattr(db, "medication_record")

        # Add the new record to the table
        medication_record_table[medication_id] = new_medication_record

        # Update the database with the modified table
        setattr(db, "medication_record", medication_record_table)

        # Return the medication_id and creation_timestamp in the required format
        return {
            "medication_id": medication_id,
            "creation_timestamp": creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def create_pet_profile(
        self,
        pet_name: str,
        species: Literal["dog", "cat", "bird", "rabbit", "hamster", "guinea_pig", "fish", "reptile", "other"],
        birth_date: str,
        owner_id: str,
        breed: str = None,
        weight_kg: float = None,
        gender: Literal["male", "female", "unknown"] = None,
        microchip_id: str = None
    ):
        """
        Create a new pet profile with basic information including species, breed, age, weight, and medical history.

        This method validates the input parameters, generates a unique pet_id, and stores the pet profile
        in the database with a creation timestamp.
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Validate required parameters are not empty
        if not pet_name or not pet_name.strip():
            raise ValueError("pet_name cannot be empty")

        if not owner_id or not owner_id.strip():
            raise ValueError("owner_id cannot be empty")

        # Validate species enum value (safety protection for enum parameter)
        valid_species = ["dog", "cat", "bird", "rabbit", "hamster", "guinea_pig", "fish", "reptile", "other"]
        if species not in valid_species:
            raise ValueError(f"Invalid species '{species}'. Must be one of: {', '.join(valid_species)}")

        # Validate gender enum value if provided (safety protection for enum parameter)
        if gender is not None:
            valid_genders = ["male", "female", "unknown"]
            if gender not in valid_genders:
                raise ValueError(f"Invalid gender '{gender}'. Must be one of: {', '.join(valid_genders)}")

        # Validate and parse birth_date format (yyyy-mm-dd)
        try:
            birth_date_obj = datetime.strptime(birth_date, "%Y-%m-%d").date()
        except ValueError as e:
            raise ValueError(f"Invalid birth_date format. Expected yyyy-mm-dd, got '{birth_date}': {str(e)}")

        # Validate birth_date is not in the future
        if birth_date_obj > datetime.now().date():
            raise ValueError("birth_date cannot be in the future")

        # Validate weight_kg if provided
        if weight_kg is not None:
            if not isinstance(weight_kg, (int, float)):
                raise ValueError("weight_kg must be a number")
            if weight_kg <= 0:
                raise ValueError("weight_kg must be greater than 0")

        # Generate unique pet_id using secure random hash
        prefix = "pet_"
        pet_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for creation_timestamp
        creation_timestamp = datetime.now()

        # Access the database
        db = self.db

        # Get the pet_profile table, initialize if it doesn't exist
        pet_profile_table = getattr(db, "pet_profile", None)
        if pet_profile_table is None:
            pet_profile_table = {}

        # Ensure pet_id is unique (very unlikely collision, but good practice)
        while pet_id in pet_profile_table:
            pet_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create new PetProfile instance
        new_pet_profile = PetProfile(
            pet_id=pet_id,
            pet_name=pet_name.strip(),
            species=species,
            breed=breed.strip() if breed else None,
            birth_date=birth_date_obj,
            weight_kg=weight_kg,
            gender=gender,
            microchip_id=microchip_id.strip() if microchip_id else None,
            owner_id=owner_id.strip(),
            creation_timestamp=creation_timestamp
        )

        # Add the new pet profile to the table
        pet_profile_table[pet_id] = new_pet_profile

        # Save the updated table back to the database
        setattr(db, "pet_profile", pet_profile_table)

        # Return the pet_id and creation_timestamp in the required format
        return {
            "pet_id": pet_id,
            "creation_timestamp": creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def create_vaccination_record(
        self,
        pet_id: str,
        vaccine_name: str,
        administration_date: str,
        next_due_date: str,
        vaccine_type: Literal["core", "non_core", "required_by_law", "optional"] = None,
        veterinarian_name: str = None,
        batch_number: str = None,
        clinic_id: str = None
    ):
        """
        Create a new vaccination record for a pet.

        This method validates input parameters, generates a unique vaccination ID,
        and stores the vaccination record in the database.

        Args:
            pet_id: Unique identifier of the pet
            vaccine_name: Name of the vaccine administered
            administration_date: Date when vaccine was administered (yyyy-mm-dd format)
            next_due_date: Date when next dose is due (yyyy-mm-dd format)
            vaccine_type: Type/category of vaccine (core/non_core/required_by_law/optional)
            veterinarian_name: Name of veterinarian who administered the vaccine
            batch_number: Batch number of the vaccine
            clinic_id: Unique identifier of the clinic where vaccination was performed

        Returns:
            dict: Contains vaccination_id and creation_timestamp

        Raises:
            ValueError: If required parameters are invalid or dates are in wrong format
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters are not empty
        if not pet_id or not isinstance(pet_id, str) or not pet_id.strip():
            raise ValueError("pet_id must be a non-empty string")

        if not vaccine_name or not isinstance(vaccine_name, str) or not vaccine_name.strip():
            raise ValueError("vaccine_name must be a non-empty string")

        if not administration_date or not isinstance(administration_date, str):
            raise ValueError("administration_date must be a non-empty string")

        if not next_due_date or not isinstance(next_due_date, str):
            raise ValueError("next_due_date must be a non-empty string")

        # Validate and parse administration_date
        try:
            admin_date_obj = datetime.strptime(administration_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError("administration_date must be in yyyy-mm-dd format")

        # Validate and parse next_due_date
        try:
            next_due_date_obj = datetime.strptime(next_due_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError("next_due_date must be in yyyy-mm-dd format")

        # Validate that next_due_date is after administration_date
        if next_due_date_obj <= admin_date_obj:
            raise ValueError("next_due_date must be after administration_date")

        # Validate vaccine_type if provided (enum constraint with safety protection)
        valid_vaccine_types = ["core", "non_core", "required_by_law", "optional"]
        if vaccine_type is not None:
            if vaccine_type not in valid_vaccine_types:
                raise ValueError(
                    f"vaccine_type must be one of {valid_vaccine_types}, got '{vaccine_type}'"
                )

        # Generate unique vaccination_id
        prefix = "vacc_"
        vaccination_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for creation_timestamp
        creation_timestamp = datetime.now()

        # Access database
        db = self.db

        # Get vaccination_record table, initialize if not exists
        vaccination_records = getattr(db, "vaccination_record", None)
        if vaccination_records is None:
            vaccination_records = {}
            setattr(db, "vaccination_record", vaccination_records)

        # Create new VaccinationRecord instance
        new_record = VaccinationRecord(
            vaccination_id=vaccination_id,
            pet_id=pet_id.strip(),
            vaccine_name=vaccine_name.strip(),
            vaccine_type=vaccine_type,
            administration_date=admin_date_obj,
            next_due_date=next_due_date_obj,
            veterinarian_name=veterinarian_name.strip() if veterinarian_name else None,
            batch_number=batch_number.strip() if batch_number else None,
            clinic_id=clinic_id.strip() if clinic_id else None,
            creation_timestamp=creation_timestamp
        )

        # Store the record in database
        vaccination_records[vaccination_id] = new_record
        setattr(db, "vaccination_record", vaccination_records)

        # Format creation_timestamp for return (yyyy-mm-dd HH:MM:SS format)
        creation_timestamp_str = creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return vaccination_id and creation_timestamp
        return {
            "vaccination_id": vaccination_id,
            "creation_timestamp": creation_timestamp_str
        }

    @is_tool()
    def get_grooming_history(self, pet_id: str, start_date: str = None, end_date: str = None):
        """
        Retrieve grooming history for a pet within an optional date range.

        Args:
            pet_id: Unique identifier of the pet
            start_date: Optional start date in yyyy-mm-dd format
            end_date: Optional end date in yyyy-mm-dd format

        Returns:
            Dictionary containing:
            - grooming_records: List of grooming record dictionaries
            - total_count: Total number of records found
            - last_grooming_date: Most recent grooming date in yyyy-mm-dd format

        Raises:
            KeyError: If pet_id is not found in any grooming records
        """
        from datetime import datetime

        # Get database instance
        db = self.db

        # Get grooming_record table from database
        grooming_table = getattr(db, "grooming_record", None)

        # Check if table exists and has data
        if grooming_table is None or len(grooming_table) == 0:
            raise KeyError(f"No grooming records found in the system")

        # Filter records by pet_id
        pet_grooming_records = []
        for record_id, record in grooming_table.items():
            if record.pet_id == pet_id:
                pet_grooming_records.append(record)

        # If no records found for this pet, raise KeyError
        if len(pet_grooming_records) == 0:
            raise KeyError(f"No grooming records found for pet_id: {pet_id}")

        # Parse date filters if provided
        start_datetime = None
        end_datetime = None

        if start_date:
            try:
                # Parse start_date string to date object for comparison
                start_datetime = datetime.strptime(start_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid start_date format. Expected yyyy-mm-dd, got: {start_date}")

        if end_date:
            try:
                # Parse end_date string to date object for comparison
                end_datetime = datetime.strptime(end_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid end_date format. Expected yyyy-mm-dd, got: {end_date}")

        # Validate date range if both provided
        if start_datetime and end_datetime and start_datetime > end_datetime:
            raise ValueError("start_date cannot be after end_date")

        # Filter records by date range if specified
        filtered_records = []
        for record in pet_grooming_records:
            # Get grooming_date from record (it's a date object in DB)
            grooming_date = record.grooming_date

            # Apply date filters
            if start_datetime and grooming_date < start_datetime:
                continue
            if end_datetime and grooming_date > end_datetime:
                continue

            filtered_records.append(record)

        # If no records match the date filter
        if len(filtered_records) == 0:
            # Return empty result structure
            return {
                "grooming_records": [],
                "total_count": 0,
                "last_grooming_date": None
            }

        # Sort records by grooming_date in descending order (most recent first)
        filtered_records.sort(key=lambda x: x.grooming_date, reverse=True)

        # Get the most recent grooming date
        last_grooming_date = filtered_records[0].grooming_date.strftime("%Y-%m-%d")

        # Build response records list
        grooming_records_list = []
        for record in filtered_records:
            # Parse services_performed string to list
            # Assuming services_performed is stored as JSON string or comma-separated
            services = []
            try:
                # Try to parse as JSON array first
                import json
                services = json.loads(record.services_performed)
            except (json.JSONDecodeError, TypeError):
                # If not JSON, try comma-separated string
                if isinstance(record.services_performed, str):
                    services = [s.strip() for s in record.services_performed.split(",") if s.strip()]
                else:
                    services = []

            # Build record dictionary with all available fields
            record_dict = {
                "grooming_record_id": record.grooming_record_id,
                "grooming_date": record.grooming_date.strftime("%Y-%m-%d"),
                "services_performed": services
            }

            # Add optional fields if they exist
            if record.groomer_name:
                record_dict["groomer_name"] = record.groomer_name
            if record.location_id:
                record_dict["location_id"] = record.location_id
            if record.duration_minutes is not None:
                record_dict["duration_minutes"] = record.duration_minutes
            if record.coat_condition:
                record_dict["coat_condition"] = record.coat_condition
            if record.notes:
                record_dict["notes"] = record.notes

            grooming_records_list.append(record_dict)

        # Return the complete grooming history
        return {
            "grooming_records": grooming_records_list,
            "total_count": len(grooming_records_list),
            "last_grooming_date": last_grooming_date
        }

    @is_tool()
    def get_active_medications(self, pet_id: str, reference_date: str = None):
        # Import required modules
        from datetime import datetime, date

        # Access the database
        db = self.db

        # Get medication_record table from database
        medication_records = getattr(db, "medication_record", None)

        # Check if medication_record table exists
        if medication_records is None:
            raise KeyError("medication_record table does not exist in the database")

        # Parse reference_date or use current date
        if reference_date:
            try:
                # Parse the reference date string in yyyy-mm-dd format
                ref_date = datetime.strptime(reference_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid reference_date format: {reference_date}. Expected format: yyyy-mm-dd")
        else:
            # Use current date if reference_date is not provided
            ref_date = date.today()

        # Initialize list to store active medications
        active_medications = []

        # Iterate through all medication records
        for med_id, medication in medication_records.items():
            # Check if the medication belongs to the specified pet
            if medication.pet_id != pet_id:
                continue

            # Check if medication is active on the reference date
            # A medication is active if:
            # 1. start_date <= reference_date
            # 2. end_date is None (ongoing) OR end_date >= reference_date

            is_started = medication.start_date <= ref_date
            is_not_ended = medication.end_date is None or medication.end_date >= ref_date

            if is_started and is_not_ended:
                # Build medication info dictionary
                med_info = {
                    "medication_id": medication.medication_id,
                    "medication_name": medication.medication_name,
                    "dosage": medication.dosage,
                    "frequency": medication.frequency,
                    "start_date": medication.start_date.strftime("%Y-%m-%d"),
                    "purpose": medication.purpose,
                    "veterinarian_name": medication.veterinarian_name,
                    "special_instructions": medication.special_instructions
                }

                # Add end_date if it exists
                if medication.end_date:
                    med_info["end_date"] = medication.end_date.strftime("%Y-%m-%d")

                active_medications.append(med_info)

        # Return the result with active medications list and total count
        return {
            "active_medications": active_medications,
            "total_count": len(active_medications)
        }

    @is_tool()
    def search_service_providers(
        self,
        service_type: Literal["veterinary", "grooming", "training", "boarding", "pet_sitting", "dog_walking", "behavioral_consultation", "emergency"],
        location: str,
        radius_km: float = None,
        min_rating: float = None,
        specializations: list = None,
        accepts_insurance: bool = None
    ):
        """
        Search for pet service providers based on criteria.

        This method searches the service_provider database for providers matching
        the specified criteria including service type, location, rating, and other filters.
        Uses fuzzy matching for location to improve search accuracy.
        """
        from thefuzz import fuzz, process
        from typing import Dict, List, Any

        # Validate service_type enum (safety protection)
        valid_service_types = ["veterinary", "grooming", "training", "boarding", 
                              "pet_sitting", "dog_walking", "behavioral_consultation", "emergency"]
        if service_type not in valid_service_types:
            raise ValueError(f"Invalid service_type: {service_type}. Must be one of {valid_service_types}")

        # Validate min_rating if provided
        if min_rating is not None:
            if not isinstance(min_rating, (int, float)) or min_rating < 0 or min_rating > 5:
                raise ValueError("min_rating must be a number between 0 and 5")

        # Validate radius_km if provided
        if radius_km is not None:
            if not isinstance(radius_km, (int, float)) or radius_km < 0:
                raise ValueError("radius_km must be a non-negative number")

        # Validate accepts_insurance if provided
        if accepts_insurance is not None and not isinstance(accepts_insurance, bool):
            raise ValueError("accepts_insurance must be a boolean value")

        # Access database
        db = self.db
        service_provider_table = getattr(db, "service_provider", None)

        # Check if service_provider table exists and has data
        if service_provider_table is None or not service_provider_table:
            return {
                "providers": [],
                "total_count": 0
            }

        # Initialize list to store matching providers
        matching_providers = []

        # Iterate through all service providers
        for provider_id, provider in service_provider_table.items():
            # Filter by service_type (exact match required)
            if provider.service_type != service_type:
                continue

            # Filter by location using fuzzy matching (for natural language text)
            # Use a threshold of 60 for location matching to allow reasonable variations
            location_similarity = fuzz.partial_ratio(location.lower(), provider.location.lower())
            if location_similarity < 60:
                continue

            # Filter by min_rating if specified
            if min_rating is not None:
                # Skip providers without rating or with rating below minimum
                if provider.rating is None or provider.rating < min_rating:
                    continue

            # Filter by accepts_insurance if specified
            if accepts_insurance is not None:
                # Skip if provider's insurance acceptance doesn't match requirement
                if provider.accepts_insurance != accepts_insurance:
                    continue

            # Filter by specializations if specified
            if specializations is not None and len(specializations) > 0:
                # Parse provider's specializations (assuming comma-separated string or None)
                provider_specializations = []
                if provider.specializations:
                    # Split by comma and strip whitespace
                    provider_specializations = [s.strip().lower() for s in provider.specializations.split(',')]

                # Check if provider has all required specializations using fuzzy matching
                has_all_specializations = True
                for required_spec in specializations:
                    required_spec_lower = required_spec.lower()
                    # Check if any provider specialization fuzzy matches the required one
                    found_match = False
                    for provider_spec in provider_specializations:
                        # Use fuzzy matching for specialization comparison
                        if fuzz.ratio(required_spec_lower, provider_spec) >= 80:
                            found_match = True
                            break

                    if not found_match:
                        has_all_specializations = False
                        break

                if not has_all_specializations:
                    continue

            # Note: radius_km filtering would require geocoding to calculate actual distances
            # Since we don't have coordinate data, we use location fuzzy matching as a proxy
            # In a real implementation, this would use geospatial calculations

            # Provider matches all criteria, add to results
            provider_info = {
                "provider_id": provider.provider_id,
                "name": provider.name,
                "service_type": provider.service_type,
                "location": provider.location,
                "phone_number": provider.phone_number,
                "email": provider.email,
                "rating": provider.rating,
                "specializations": provider.specializations,
                "accepts_insurance": provider.accepts_insurance,
                "available_hours": provider.available_hours
            }
            matching_providers.append(provider_info)

        # Sort providers by rating (descending) if rating exists
        # Providers without rating are placed at the end
        matching_providers.sort(
            key=lambda p: (p["rating"] is not None, p["rating"] if p["rating"] is not None else 0),
            reverse=True
        )

        # Return results
        return {
            "providers": matching_providers,
            "total_count": len(matching_providers)
        }

    @is_tool()
    def create_diet_log(self, pet_id: str, log_date: str, meal_time: str, food_items: list, portion_grams: float, calories: float = None, water_intake_ml: float = None, notes: str = None):
        """
        Create a daily diet log entry for a pet

        This method creates a new diet log entry in the database with the provided meal information.
        It validates the input parameters, generates a unique diet_log_id, and stores the entry
        with the current timestamp.

        Args:
            pet_id: Unique identifier of the pet
            log_date: Date of the diet log in yyyy-mm-dd format
            meal_time: Time of the meal in HH:MM:SS format
            food_items: List of food items consumed
            portion_grams: Total portion size in grams
            calories: Optional estimated calories consumed
            water_intake_ml: Optional water intake in milliliters
            notes: Optional additional notes about the meal

        Returns:
            dict: Contains diet_log_id and creation_timestamp

        Raises:
            ValueError: If required parameters are invalid or missing
        """
        import secrets
        import hashlib
        from datetime import datetime
        import json

        # Validate required parameters
        if not pet_id or not isinstance(pet_id, str):
            raise ValueError("pet_id must be a non-empty string")

        if not log_date or not isinstance(log_date, str):
            raise ValueError("log_date must be a non-empty string in yyyy-mm-dd format")

        if not meal_time or not isinstance(meal_time, str):
            raise ValueError("meal_time must be a non-empty string in HH:MM:SS format")

        if not food_items or not isinstance(food_items, list) or len(food_items) == 0:
            raise ValueError("food_items must be a non-empty list")

        if portion_grams is None or not isinstance(portion_grams, (int, float)) or portion_grams <= 0:
            raise ValueError("portion_grams must be a positive number")

        # Validate and parse log_date
        try:
            parsed_log_date = datetime.strptime(log_date, "%Y-%m-%d").date()
        except ValueError as e:
            raise ValueError(f"log_date must be in yyyy-mm-dd format: {str(e)}")

        # Validate and parse meal_time
        try:
            parsed_meal_time = datetime.strptime(meal_time, "%H:%M:%S").time()
        except ValueError as e:
            raise ValueError(f"meal_time must be in HH:MM:SS format: {str(e)}")

        # Validate optional numeric parameters
        if calories is not None and (not isinstance(calories, (int, float)) or calories < 0):
            raise ValueError("calories must be a non-negative number if provided")

        if water_intake_ml is not None and (not isinstance(water_intake_ml, (int, float)) or water_intake_ml < 0):
            raise ValueError("water_intake_ml must be a non-negative number if provided")

        # Validate notes if provided
        if notes is not None and not isinstance(notes, str):
            raise ValueError("notes must be a string if provided")

        # Generate unique diet_log_id
        prefix = "dietlog_"
        diet_log_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for creation
        creation_timestamp = datetime.now()

        # Convert food_items list to JSON string for storage
        food_items_str = json.dumps(food_items)

        # Access the database
        db = self.db

        # Initialize diet_log table if it doesn't exist
        if getattr(db, "diet_log", None) is None:
            setattr(db, "diet_log", {})

        # Get the diet_log table
        diet_log_table = getattr(db, "diet_log")

        # Create new DietLog entry
        new_diet_log = DietLog(
            diet_log_id=diet_log_id,
            pet_id=pet_id,
            log_date=parsed_log_date,
            meal_time=parsed_meal_time,
            food_items=food_items_str,
            portion_grams=float(portion_grams),
            calories=float(calories) if calories is not None else None,
            water_intake_ml=float(water_intake_ml) if water_intake_ml is not None else None,
            notes=notes,
            creation_timestamp=creation_timestamp
        )

        # Store the new diet log entry in the database
        diet_log_table[diet_log_id] = new_diet_log
        setattr(db, "diet_log", diet_log_table)

        # Format creation_timestamp as string in yyyy-mm-dd HH:MM:SS format
        creation_timestamp_str = creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "diet_log_id": diet_log_id,
            "creation_timestamp": creation_timestamp_str
        }

    @is_tool()
    def calculate_household_care_cost(
        self,
        household_id: str,
        start_date: str,
        end_date: str,
        cost_categories: Optional[List[Literal["veterinary", "medication", "grooming", "food", "training", "boarding", "insurance", "supplies", "emergency"]]] = None
    ):
        """
        Calculate total pet care costs for a household over a specified period.

        This method aggregates costs from various care categories (veterinary, medication, grooming, insurance)
        for all pets in a household within the given date range.
        """
        from datetime import datetime
        from collections import defaultdict

        # Access database
        db = self.db

        # Validate household exists
        household_table = getattr(db, "household", None)
        if household_table is None or household_id not in household_table:
            raise KeyError(f"Household with ID '{household_id}' not found")

        # Parse date strings to datetime objects for comparison
        try:
            start_dt = datetime.strptime(start_date, "%Y-%m-%d")
            end_dt = datetime.strptime(end_date, "%Y-%m-%d")
        except ValueError as e:
            raise ValueError(f"Invalid date format. Expected yyyy-mm-dd: {e}")

        # Validate date range
        if start_dt > end_dt:
            raise ValueError("Start date must be before or equal to end date")

        # Validate cost_categories enum values if provided
        valid_categories = ["veterinary", "medication", "grooming", "food", "training", "boarding", "insurance", "supplies", "emergency"]
        if cost_categories is not None:
            for category in cost_categories:
                if category not in valid_categories:
                    raise ValueError(f"Invalid cost category: '{category}'. Must be one of {valid_categories}")
            # Filter to only include categories we have data for
            available_categories = ["veterinary", "medication", "grooming", "insurance"]
            cost_categories = [cat for cat in cost_categories if cat in available_categories]
        else:
            # If no cost_categories specified, include all available categories
            cost_categories = ["veterinary", "medication", "grooming", "insurance"]

        # Get all pets in the household
        household_pet_table = getattr(db, "household_pet", None)
        if household_pet_table is None:
            household_pet_table = {}

        pet_ids = [hp.pet_id for hp in household_pet_table.values() if hp.household_id == household_id]

        if not pet_ids:
            # No pets in household, return zero costs
            return {
                "total_cost": 0.0,
                "cost_by_category": {},
                "cost_by_pet": [],
                "average_monthly_cost": 0.0
            }

        # Initialize cost tracking structures
        cost_by_category = defaultdict(float)
        cost_by_pet_dict = {pet_id: {"pet_id": pet_id, "pet_name": None, "total_cost": 0.0} for pet_id in pet_ids}

        # Calculate veterinary costs from health checkup records
        if "veterinary" in cost_categories:
            checkup_table = getattr(db, "health_checkup_record", None)
            if checkup_table is not None:
                for record in checkup_table.values():
                    if record.pet_id in pet_ids:
                        # Convert checkup_date (date object) to datetime for comparison
                        checkup_dt = datetime.combine(record.checkup_date, datetime.min.time())
                        if start_dt <= checkup_dt <= end_dt:
                            # Estimate veterinary costs based on checkup type
                            # These are reasonable estimates for different checkup types
                            cost_mapping = {
                                "routine": 150.0,
                                "emergency": 500.0,
                                "follow_up": 100.0,
                                "pre_surgery": 200.0,
                                "post_surgery": 150.0,
                                "dental": 300.0,
                                "specialized": 400.0
                            }
                            cost = cost_mapping.get(record.checkup_type, 150.0)
                            cost_by_category["veterinary"] += cost
                            cost_by_pet_dict[record.pet_id]["total_cost"] += cost

        # Calculate medication costs from medication records
        if "medication" in cost_categories:
            medication_table = getattr(db, "medication_record", None)
            if medication_table is not None:
                for record in medication_table.values():
                    if record.pet_id in pet_ids:
                        # Check if medication period overlaps with our date range
                        med_start_dt = datetime.combine(record.start_date, datetime.min.time())
                        med_end_dt = datetime.combine(record.end_date, datetime.min.time()) if record.end_date else end_dt

                        # Check for overlap between [start_dt, end_dt] and [med_start_dt, med_end_dt]
                        overlap_start = max(start_dt, med_start_dt)
                        overlap_end = min(end_dt, med_end_dt)

                        if overlap_start <= overlap_end:
                            # Calculate days of overlap
                            overlap_days = (overlap_end - overlap_start).days + 1

                            # Estimate medication cost based on duration
                            # Average medication cost: $50 per month (approximately $1.67 per day)
                            daily_cost = 1.67
                            cost = overlap_days * daily_cost

                            cost_by_category["medication"] += cost
                            cost_by_pet_dict[record.pet_id]["total_cost"] += cost

        # Calculate grooming costs from grooming records
        if "grooming" in cost_categories:
            grooming_table = getattr(db, "grooming_record", None)
            if grooming_table is not None:
                for record in grooming_table.values():
                    if record.pet_id in pet_ids:
                        grooming_dt = datetime.combine(record.grooming_date, datetime.min.time())
                        if start_dt <= grooming_dt <= end_dt:
                            # Estimate grooming cost based on duration
                            # Average grooming cost: $60 base + $1 per minute
                            base_cost = 60.0
                            duration_cost = record.duration_minutes if record.duration_minutes else 0
                            cost = base_cost + duration_cost

                            cost_by_category["grooming"] += cost
                            cost_by_pet_dict[record.pet_id]["total_cost"] += cost

        # Calculate insurance costs from insurance claims
        if "insurance" in cost_categories:
            insurance_table = getattr(db, "insurance_claim", None)
            if insurance_table is not None:
                for record in insurance_table.values():
                    if record.pet_id in pet_ids:
                        claim_dt = datetime.combine(record.claim_date, datetime.min.time())
                        if start_dt <= claim_dt <= end_dt:
                            # Use total_cost from the claim
                            # This represents the out-of-pocket cost for the pet owner
                            cost = record.total_cost

                            # If claim is approved/partially approved, subtract the approved amount
                            if record.approved_amount and record.status in ["approved", "partially_approved", "paid"]:
                                cost = max(0, cost - record.approved_amount)

                            cost_by_category["insurance"] += cost
                            cost_by_pet_dict[record.pet_id]["total_cost"] += cost

        # Calculate total cost
        total_cost = sum(cost_by_category.values())

        # Convert cost_by_pet_dict to list format
        cost_by_pet = [pet_data for pet_data in cost_by_pet_dict.values() if pet_data["total_cost"] > 0]

        # Calculate average monthly cost
        # Calculate number of months in the period
        days_in_period = (end_dt - start_dt).days + 1
        months_in_period = max(1, days_in_period / 30.0)  # Use at least 1 month to avoid division issues

        average_monthly_cost = round(total_cost / months_in_period, 2)

        # Round all costs to 2 decimal places for consistency
        total_cost = round(total_cost, 2)
        cost_by_category = {k: round(v, 2) for k, v in cost_by_category.items()}
        for pet_data in cost_by_pet:
            pet_data["total_cost"] = round(pet_data["total_cost"], 2)

        return {
            "total_cost": total_cost,
            "cost_by_category": dict(cost_by_category),
            "cost_by_pet": cost_by_pet,
            "average_monthly_cost": average_monthly_cost
        }

    @is_tool()
    def generate_vaccination_certificate(
        self,
        pet_id: str,
        vaccination_ids: List[str],
        certificate_purpose: Literal["travel", "boarding", "training", "general", "legal_requirement"],
        issue_date: str = None
    ) -> dict:
        """
        Generate an official vaccination certificate for a pet.

        This method creates a vaccination certificate that includes specified vaccination records
        for a pet. The certificate is stored in the database and includes a unique QR code for
        verification purposes.

        Args:
            pet_id: Unique identifier of the pet
            vaccination_ids: List of vaccination record identifiers to include in the certificate
            certificate_purpose: Purpose of the certificate (must be one of the enum values)
            issue_date: Date of certificate issuance in yyyy-mm-dd format (defaults to current date)

        Returns:
            Dictionary containing:
                - certificate_id: Unique identifier for the certificate
                - certificate_number: Official certificate number
                - issue_timestamp: Timestamp when certificate was issued
                - valid_until: Certificate validity end date
                - qr_code: QR code for certificate verification

        Raises:
            KeyError: If pet_id doesn't exist or any vaccination_id doesn't exist
        """
        from datetime import datetime, timedelta
        import secrets
        import hashlib

        # Access database
        db = self.db

        # Validate pet_id exists
        pet_profiles = getattr(db, "pet_profile", None)
        if pet_profiles is None or pet_id not in pet_profiles:
            raise KeyError(f"Pet with id '{pet_id}' does not exist")

        # Validate all vaccination_ids exist and belong to the pet
        vaccination_records = getattr(db, "vaccination_record", None)
        if vaccination_records is None:
            raise KeyError("No vaccination records found in database")

        validated_vaccinations = []
        for vacc_id in vaccination_ids:
            if vacc_id not in vaccination_records:
                raise KeyError(f"Vaccination record with id '{vacc_id}' does not exist")

            vaccination = vaccination_records[vacc_id]
            # Verify vaccination belongs to the specified pet
            if vaccination.pet_id != pet_id:
                raise KeyError(f"Vaccination record '{vacc_id}' does not belong to pet '{pet_id}'")

            validated_vaccinations.append(vaccination)

        # Parse issue_date or use current date
        if issue_date:
            try:
                issue_date_obj = datetime.strptime(issue_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid issue_date format: '{issue_date}'. Expected format: yyyy-mm-dd")
        else:
            issue_date_obj = datetime.now().date()

        # Generate certificate_id using secure random hash
        certificate_id = "cert_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Generate certificate_number with timestamp-based sequential format
        current_timestamp = datetime.now()
        certificate_number = f"VACC-CERT-{current_timestamp.year}-{hashlib.sha256(secrets.token_bytes(16)).hexdigest()[:8].upper()}"

        # Calculate validity period based on the earliest next_due_date among vaccinations
        # Certificate is valid until the earliest vaccine needs renewal
        earliest_next_due = min(vacc.next_due_date for vacc in validated_vaccinations)
        valid_until_date = earliest_next_due

        # Generate QR code data string containing certificate verification information
        qr_data_components = [
            f"CERT:{certificate_number}",
            f"PET:{pet_id}",
            f"ISSUED:{issue_date_obj.isoformat()}",
            f"VALID:{valid_until_date.isoformat()}",
            f"PURPOSE:{certificate_purpose}"
        ]
        qr_code = "|".join(qr_data_components)

        # Create issue timestamp
        issue_timestamp = datetime.now()

        # Create VaccinationCertificate instance
        new_certificate = VaccinationCertificate(
            certificate_id=certificate_id,
            pet_id=pet_id,
            certificate_number=certificate_number,
            certificate_purpose=certificate_purpose,
            issue_date=issue_date_obj,
            valid_until=valid_until_date,
            qr_code=qr_code,
            issue_timestamp=issue_timestamp
        )

        # Store certificate in database
        vaccination_certificates = getattr(db, "vaccination_certificate", {})
        vaccination_certificates[certificate_id] = new_certificate
        setattr(db, "vaccination_certificate", vaccination_certificates)

        # Create CertificateVaccination associations for each vaccination record
        certificate_vaccinations = getattr(db, "certificate_vaccination", {})
        for vacc_id in vaccination_ids:
            # Generate unique association id
            assoc_id = "cv_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Create association record
            cert_vacc_assoc = CertificateVaccination(
                certificate_vaccination_id=assoc_id,
                certificate_id=certificate_id,
                vaccination_id=vacc_id
            )

            certificate_vaccinations[assoc_id] = cert_vacc_assoc

        setattr(db, "certificate_vaccination", certificate_vaccinations)

        # Return certificate details in required format
        return {
            "certificate_id": certificate_id,
            "certificate_number": certificate_number,
            "issue_timestamp": issue_timestamp.strftime("%Y-%m-%d %H:%M:%S"),
            "valid_until": valid_until_date.strftime("%Y-%m-%d"),
            "qr_code": qr_code
        }

    @is_tool()
    def rate_service_provider(self, provider_id: str, pet_id: str, rating: float, appointment_id: str = None, review_text: str = None, review_date: str = None, would_recommend: bool = None):
        """
        Submit a rating and review for a service provider.

        This method creates a new provider review entry in the database with the given
        rating and optional review details. It validates the rating score and generates
        a unique review ID and submission timestamp.

        Args:
            provider_id: Unique identifier of the service provider being reviewed
            pet_id: Unique identifier of the pet that received the service
            rating: Rating score from 1 to 5 (can be decimal, e.g., 4.5)
            appointment_id: Optional unique identifier of the appointment
            review_text: Optional written review of the service
            review_date: Optional date of the review in yyyy-mm-dd format
            would_recommend: Optional boolean indicating if reviewer would recommend this provider

        Returns:
            dict: Contains review_id and submission_timestamp
                - review_id: Unique identifier for the newly created review
                - submission_timestamp: Timestamp when review was submitted (yyyy-mm-dd HH:MM:SS format)

        Raises:
            ValueError: If rating is not between 1 and 5, or if review_date format is invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate rating score (must be between 1 and 5)
        if not (1 <= rating <= 5):
            raise ValueError(f"Rating must be between 1 and 5, got {rating}")

        # Parse review_date if provided
        parsed_review_date = None
        if review_date is not None:
            try:
                # Parse the date string to ensure it's in correct format
                parsed_review_date = datetime.strptime(review_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid review_date format. Expected yyyy-mm-dd, got {review_date}")

        # Generate unique review_id using secure random hash
        review_id = "review_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Generate submission timestamp (current time)
        submission_timestamp = datetime.now()

        # Access the database
        db = self.db

        # Get the provider_review table (initialize if None)
        provider_review_table = getattr(db, "provider_review", None)
        if provider_review_table is None:
            provider_review_table = {}

        # Create new ProviderReview instance
        new_review = ProviderReview(
            review_id=review_id,
            provider_id=provider_id,
            pet_id=pet_id,
            appointment_id=appointment_id,
            rating=rating,
            review_text=review_text,
            review_date=parsed_review_date,
            would_recommend=would_recommend,
            submission_timestamp=submission_timestamp
        )

        # Add the new review to the table
        provider_review_table[review_id] = new_review

        # Save the updated table back to the database
        setattr(db, "provider_review", provider_review_table)

        # Return the review_id and submission_timestamp in the required format
        return {
            "review_id": review_id,
            "submission_timestamp": submission_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def compare_pet_health_metrics(self, pet_ids: List[str], metrics: List[Literal["weight", "exercise_frequency", "vaccination_status", "checkup_frequency", "medication_count", "behavior_issues"]], reference_date: str = None):
        """
        Compare health metrics between multiple pets in a household.

        This method analyzes and compares various health metrics across multiple pets,
        providing both detailed metric comparisons and overall health summaries.
        """
        from datetime import datetime, timedelta

        # Access the database
        db = self.db

        # Validate that all pet_ids exist in the database
        pet_profile_table = getattr(db, "pet_profile", None)
        if pet_profile_table is None:
            raise KeyError("pet_profile table not found in database")

        for pet_id in pet_ids:
            if pet_id not in pet_profile_table:
                raise KeyError(f"Pet with ID '{pet_id}' not found in pet_profile table")

        # Parse reference_date if provided, otherwise use current date
        if reference_date:
            try:
                ref_date = datetime.strptime(reference_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid reference_date format: {reference_date}. Expected format: yyyy-mm-dd")
        else:
            ref_date = datetime.now().date()

        # Validate metrics parameter - ensure all values are from the allowed enum
        allowed_metrics = ["weight", "exercise_frequency", "vaccination_status", "checkup_frequency", "medication_count", "behavior_issues"]
        for metric in metrics:
            if metric not in allowed_metrics:
                raise ValueError(f"Invalid metric '{metric}'. Allowed values: {allowed_metrics}")

        # Initialize results structure
        comparison_results = []
        health_summary = []

        # Process each metric
        for metric in metrics:
            metric_result = {"metric": metric}

            if metric == "weight":
                # Get the most recent weight for each pet (from weight_history or pet_profile)
                weight_history_table = getattr(db, "weight_history", None)

                for pet_id in pet_ids:
                    pet_weight = None

                    # Try to get weight from weight_history first
                    if weight_history_table:
                        # Find the most recent weight record on or before reference_date
                        relevant_records = [
                            record for record in weight_history_table.values()
                            if record.pet_id == pet_id and record.measurement_date <= ref_date
                        ]
                        if relevant_records:
                            latest_record = max(relevant_records, key=lambda r: r.measurement_date)
                            pet_weight = latest_record.weight_kg

                    # Fallback to pet_profile if no weight_history found
                    if pet_weight is None:
                        pet_profile = pet_profile_table.get(pet_id)
                        if pet_profile and pet_profile.weight_kg is not None:
                            pet_weight = pet_profile.weight_kg

                    metric_result[pet_id] = pet_weight

            elif metric == "exercise_frequency":
                # Calculate exercise frequency (number of exercise sessions in last 30 days)
                exercise_log_table = getattr(db, "exercise_log", None)

                for pet_id in pet_ids:
                    exercise_count = 0

                    if exercise_log_table:
                        # Count exercise logs in the 30 days before reference_date
                        start_date = ref_date - timedelta(days=30)
                        exercise_count = sum(
                            1 for record in exercise_log_table.values()
                            if record.pet_id == pet_id and start_date <= record.exercise_date <= ref_date
                        )

                    metric_result[pet_id] = exercise_count

            elif metric == "vaccination_status":
                # Check vaccination status (up-to-date, due, overdue)
                vaccination_record_table = getattr(db, "vaccination_record", None)

                for pet_id in pet_ids:
                    status = "unknown"

                    if vaccination_record_table:
                        pet_vaccinations = [
                            record for record in vaccination_record_table.values()
                            if record.pet_id == pet_id
                        ]

                        if pet_vaccinations:
                            # Check if any vaccination is overdue (next_due_date before reference_date)
                            overdue = any(record.next_due_date < ref_date for record in pet_vaccinations)
                            # Check if any vaccination is due soon (within 30 days)
                            due_soon = any(
                                ref_date <= record.next_due_date <= ref_date + timedelta(days=30)
                                for record in pet_vaccinations
                            )

                            if overdue:
                                status = "overdue"
                            elif due_soon:
                                status = "due"
                            else:
                                status = "up_to_date"
                        else:
                            status = "no_records"

                    metric_result[pet_id] = status

            elif metric == "checkup_frequency":
                # Calculate checkup frequency (number of checkups in last 12 months)
                health_checkup_record_table = getattr(db, "health_checkup_record", None)

                for pet_id in pet_ids:
                    checkup_count = 0

                    if health_checkup_record_table:
                        # Count checkups in the 12 months before reference_date
                        start_date = ref_date - timedelta(days=365)
                        checkup_count = sum(
                            1 for record in health_checkup_record_table.values()
                            if record.pet_id == pet_id and start_date <= record.checkup_date <= ref_date
                        )

                    metric_result[pet_id] = checkup_count

            elif metric == "medication_count":
                # Count active medications on reference_date
                medication_record_table = getattr(db, "medication_record", None)

                for pet_id in pet_ids:
                    active_medication_count = 0

                    if medication_record_table:
                        # Count medications that are active on reference_date
                        active_medication_count = sum(
                            1 for record in medication_record_table.values()
                            if record.pet_id == pet_id 
                            and record.start_date <= ref_date
                            and (record.end_date is None or record.end_date >= ref_date)
                        )

                    metric_result[pet_id] = active_medication_count

            elif metric == "behavior_issues":
                # Count behavior issues in last 90 days (moderate or severe)
                behavior_log_table = getattr(db, "behavior_log", None)

                for pet_id in pet_ids:
                    issue_count = 0

                    if behavior_log_table:
                        # Count behavior logs in the 90 days before reference_date with moderate or severe severity
                        start_datetime = datetime.combine(ref_date - timedelta(days=90), datetime.min.time())
                        end_datetime = datetime.combine(ref_date, datetime.max.time())

                        issue_count = sum(
                            1 for record in behavior_log_table.values()
                            if record.pet_id == pet_id 
                            and start_datetime <= record.observation_datetime <= end_datetime
                            and record.severity in ["moderate", "severe"]
                        )

                    metric_result[pet_id] = issue_count

            comparison_results.append(metric_result)

        # Generate health summary for each pet
        for pet_id in pet_ids:
            # Calculate overall health score based on metrics
            health_indicators = {
                "weight_ok": True,
                "exercise_ok": True,
                "vaccination_ok": True,
                "checkup_ok": True,
                "medication_ok": True,
                "behavior_ok": True
            }

            for result in comparison_results:
                metric_name = result["metric"]
                value = result.get(pet_id)

                if metric_name == "vaccination_status":
                    # Vaccination status should be "up_to_date"
                    if value in ["overdue", "no_records"]:
                        health_indicators["vaccination_ok"] = False

                elif metric_name == "exercise_frequency":
                    # Exercise frequency should be at least 8 times per month (roughly 2 per week)
                    if value is not None and value < 8:
                        health_indicators["exercise_ok"] = False

                elif metric_name == "checkup_frequency":
                    # Should have at least 1 checkup per year
                    if value is not None and value < 1:
                        health_indicators["checkup_ok"] = False

                elif metric_name == "behavior_issues":
                    # Behavior issues should be minimal (less than 5 in 90 days)
                    if value is not None and value >= 5:
                        health_indicators["behavior_ok"] = False

            # Determine overall health status
            issues_count = sum(1 for ok in health_indicators.values() if not ok)

            if issues_count == 0:
                overall_health = "excellent"
            elif issues_count <= 1:
                overall_health = "good"
            elif issues_count <= 3:
                overall_health = "fair"
            else:
                overall_health = "needs_attention"

            health_summary.append({
                "pet_id": pet_id,
                "overall_health": overall_health
            })

        return {
            "comparison_results": comparison_results,
            "health_summary": health_summary
        }

    @is_tool()
    def create_emergency_contact(
        self,
        pet_id: str,
        contact_type: Literal[
            "primary_veterinarian",
            "emergency_clinic",
            "pet_sitter",
            "family_member",
            "friend",
            "neighbor",
            "pet_poison_control"
        ],
        contact_name: str,
        phone_number: str,
        alternate_phone: str = None,
        email: str = None,
        address: str = None,
        available_hours: str = None,
        notes: str = None
    ):
        """
        Create an emergency contact record for a pet.

        This method creates a new emergency contact entry in the database with the provided
        contact information. It validates required fields, generates a unique emergency contact ID,
        and stores the contact with a creation timestamp.

        Args:
            pet_id: Unique identifier of the pet
            contact_type: Type of emergency contact (must be one of the enum values)
            contact_name: Name of the contact person or organization
            phone_number: Primary phone number
            alternate_phone: Optional alternate phone number
            email: Optional email address
            address: Optional physical address
            available_hours: Optional hours of availability
            notes: Optional additional notes about the contact

        Returns:
            dict: Contains emergency_contact_id and creation_timestamp

        Raises:
            ValueError: If required fields are missing or invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters
        if not pet_id or not isinstance(pet_id, str) or pet_id.strip() == "":
            raise ValueError("pet_id is required and must be a non-empty string")

        if not contact_name or not isinstance(contact_name, str) or contact_name.strip() == "":
            raise ValueError("contact_name is required and must be a non-empty string")

        if not phone_number or not isinstance(phone_number, str) or phone_number.strip() == "":
            raise ValueError("phone_number is required and must be a non-empty string")

        # Validate contact_type enum (additional safety check)
        valid_contact_types = [
            "primary_veterinarian",
            "emergency_clinic",
            "pet_sitter",
            "family_member",
            "friend",
            "neighbor",
            "pet_poison_control"
        ]
        if contact_type not in valid_contact_types:
            raise ValueError(
                f"contact_type must be one of {valid_contact_types}, got '{contact_type}'"
            )

        # Access the database
        db = self.db

        # Get the emergency_contact table from database
        emergency_contact_table = getattr(db, "emergency_contact", None)
        if emergency_contact_table is None:
            # Initialize the table if it doesn't exist
            emergency_contact_table = {}
            setattr(db, "emergency_contact", emergency_contact_table)

        # Generate unique emergency contact ID
        prefix = "emerg_"
        emergency_contact_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Ensure the generated ID is unique
        while emergency_contact_id in emergency_contact_table:
            emergency_contact_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        creation_timestamp = datetime.now()

        # Create the EmergencyContact instance
        new_emergency_contact = EmergencyContact(
            emergency_contact_id=emergency_contact_id,
            pet_id=pet_id.strip(),
            contact_type=contact_type,
            contact_name=contact_name.strip(),
            phone_number=phone_number.strip(),
            alternate_phone=alternate_phone.strip() if alternate_phone else None,
            email=email.strip() if email else None,
            address=address.strip() if address else None,
            available_hours=available_hours.strip() if available_hours else None,
            notes=notes.strip() if notes else None,
            creation_timestamp=creation_timestamp
        )

        # Add the new emergency contact to the table
        emergency_contact_table[emergency_contact_id] = new_emergency_contact

        # Update the database with the modified table
        setattr(db, "emergency_contact", emergency_contact_table)

        # Format the creation timestamp as string in "yyyy-mm-dd HH:MM:SS" format
        creation_timestamp_str = creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "emergency_contact_id": emergency_contact_id,
            "creation_timestamp": creation_timestamp_str
        }

    @is_tool()
    def create_allergy_record(
        self,
        pet_id: str,
        allergen: str,
        allergy_type: Literal["food", "environmental", "medication", "contact", "insect", "other"],
        severity: Literal["mild", "moderate", "severe", "life_threatening"],
        symptoms: Optional[List[str]] = None,
        diagnosed_date: Optional[str] = None,
        veterinarian_name: Optional[str] = None,
        notes: Optional[str] = None
    ) -> Dict[str, str]:
        """
        Create a new allergy record for a pet.

        This method creates a new allergy record with the provided information and stores it
        in the allergy_record database table. It generates a unique allergy_id and records
        the creation timestamp.

        Args:
            pet_id: Unique identifier of the pet
            allergen: Name of the allergen (e.g., "Chicken", "Pollen")
            allergy_type: Type of allergy, must be one of the enumerated values
            severity: Severity level of the allergy, must be one of the enumerated values
            symptoms: Optional list of symptoms associated with the allergy
            diagnosed_date: Optional date when the allergy was diagnosed (format: yyyy-mm-dd)
            veterinarian_name: Optional name of the veterinarian who diagnosed the allergy
            notes: Optional additional notes about the allergy

        Returns:
            Dictionary containing:
                - allergy_id: Unique identifier for the created allergy record
                - creation_timestamp: Timestamp when the record was created (format: yyyy-mm-dd HH:MM:SS)

        Raises:
            ValueError: If required parameters are invalid or missing
        """
        from datetime import datetime, date
        import secrets
        import hashlib

        # Validate required parameters
        if not pet_id or not isinstance(pet_id, str) or not pet_id.strip():
            raise ValueError("pet_id must be a non-empty string")

        if not allergen or not isinstance(allergen, str) or not allergen.strip():
            raise ValueError("allergen must be a non-empty string")

        # Validate allergy_type enum (with safety protection)
        valid_allergy_types = ["food", "environmental", "medication", "contact", "insect", "other"]
        if allergy_type not in valid_allergy_types:
            raise ValueError(f"allergy_type must be one of {valid_allergy_types}, got: {allergy_type}")

        # Validate severity enum (with safety protection)
        valid_severities = ["mild", "moderate", "severe", "life_threatening"]
        if severity not in valid_severities:
            raise ValueError(f"severity must be one of {valid_severities}, got: {severity}")

        # Validate and parse diagnosed_date if provided
        diagnosed_date_obj = None
        if diagnosed_date is not None:
            if not isinstance(diagnosed_date, str):
                raise ValueError("diagnosed_date must be a string in yyyy-mm-dd format")
            try:
                diagnosed_date_obj = datetime.strptime(diagnosed_date.strip(), "%Y-%m-%d").date()
            except ValueError as e:
                raise ValueError(f"diagnosed_date must be in yyyy-mm-dd format, got: {diagnosed_date}") from e

        # Validate symptoms if provided
        if symptoms is not None:
            if not isinstance(symptoms, list):
                raise ValueError("symptoms must be a list of strings")
            if not all(isinstance(s, str) for s in symptoms):
                raise ValueError("all symptoms must be strings")

        # Validate veterinarian_name if provided
        if veterinarian_name is not None and not isinstance(veterinarian_name, str):
            raise ValueError("veterinarian_name must be a string")

        # Validate notes if provided
        if notes is not None and not isinstance(notes, str):
            raise ValueError("notes must be a string")

        # Generate unique allergy_id
        prefix = "allergy_"
        allergy_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        creation_timestamp = datetime.now()

        # Convert symptoms list to JSON string for storage (database stores as Optional[str])
        symptoms_str = None
        if symptoms is not None:
            symptoms_str = json.dumps(symptoms)

        # Access the database
        db = self.db

        # Get current allergy_record table (initialize if None)
        allergy_record_table = getattr(db, "allergy_record", None)
        if allergy_record_table is None:
            allergy_record_table = {}

        # Create new AllergyRecord instance
        new_allergy_record = AllergyRecord(
            allergy_id=allergy_id,
            pet_id=pet_id,
            allergen=allergen,
            allergy_type=allergy_type,
            severity=severity,
            symptoms=symptoms_str,
            diagnosed_date=diagnosed_date_obj,
            veterinarian_name=veterinarian_name,
            notes=notes,
            creation_timestamp=creation_timestamp
        )

        # Add the new record to the table
        allergy_record_table[allergy_id] = new_allergy_record

        # Update the database
        setattr(db, "allergy_record", allergy_record_table)

        # Format creation_timestamp for return value
        creation_timestamp_str = creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "allergy_id": allergy_id,
            "creation_timestamp": creation_timestamp_str
        }

    @is_tool()
    def create_exercise_log(
        self,
        pet_id: str,
        exercise_date: str,
        exercise_type: Literal["walk", "run", "swim", "play", "training", "agility", "fetch", "hiking", "other"],
        duration_minutes: int,
        distance_km: Optional[float] = None,
        intensity: Optional[Literal["low", "moderate", "high", "very_high"]] = None,
        calories_burned: Optional[float] = None,
        notes: Optional[str] = None
    ) -> dict:
        # Import necessary modules for ID generation and datetime handling
        import secrets
        import hashlib
        from datetime import datetime

        # Validate required parameters
        if not pet_id or not isinstance(pet_id, str):
            raise ValueError("pet_id must be a non-empty string")

        if not exercise_date or not isinstance(exercise_date, str):
            raise ValueError("exercise_date must be a non-empty string")

        if not exercise_type or not isinstance(exercise_type, str):
            raise ValueError("exercise_type must be a non-empty string")

        if not isinstance(duration_minutes, int) or duration_minutes <= 0:
            raise ValueError("duration_minutes must be a positive integer")

        # Validate exercise_date format (yyyy-mm-dd)
        try:
            parsed_date = datetime.strptime(exercise_date, "%Y-%m-%d")
        except ValueError:
            raise ValueError("exercise_date must be in yyyy-mm-dd format")

        # Validate exercise_type enum (with safety protection)
        valid_exercise_types = ["walk", "run", "swim", "play", "training", "agility", "fetch", "hiking", "other"]
        if exercise_type not in valid_exercise_types:
            raise ValueError(f"exercise_type must be one of {valid_exercise_types}, got '{exercise_type}'")

        # Validate optional parameters
        if distance_km is not None:
            if not isinstance(distance_km, (int, float)) or distance_km < 0:
                raise ValueError("distance_km must be a non-negative number")

        if intensity is not None:
            valid_intensities = ["low", "moderate", "high", "very_high"]
            if intensity not in valid_intensities:
                raise ValueError(f"intensity must be one of {valid_intensities}, got '{intensity}'")

        if calories_burned is not None:
            if not isinstance(calories_burned, (int, float)) or calories_burned < 0:
                raise ValueError("calories_burned must be a non-negative number")

        if notes is not None and not isinstance(notes, str):
            raise ValueError("notes must be a string")

        # Generate unique exercise_log_id
        prefix = "exlog_"
        exercise_log_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for creation_timestamp
        creation_timestamp = datetime.now()

        # Access the database
        db = self.db

        # Initialize exercise_log table if it doesn't exist
        if getattr(db, "exercise_log", None) is None:
            setattr(db, "exercise_log", {})

        # Get the exercise_log table
        exercise_log_table = getattr(db, "exercise_log")

        # Convert exercise_date string to datetime object for storage
        exercise_date_obj = parsed_date

        # Create new exercise log entry
        new_exercise_log = ExerciseLog(
            exercise_log_id=exercise_log_id,
            pet_id=pet_id,
            exercise_date=exercise_date_obj,
            exercise_type=exercise_type,
            duration_minutes=duration_minutes,
            distance_km=distance_km,
            intensity=intensity,
            calories_burned=calories_burned,
            notes=notes,
            creation_timestamp=creation_timestamp
        )

        # Store the exercise log in the database
        exercise_log_table[exercise_log_id] = new_exercise_log
        setattr(db, "exercise_log", exercise_log_table)

        # Return the result with formatted timestamp
        return {
            "exercise_log_id": exercise_log_id,
            "creation_timestamp": creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def get_emergency_contacts(self, pet_id: str):
        # Get the database instance
        db = self.db

        # Retrieve the emergency_contact table from the database
        emergency_contact_table = getattr(db, "emergency_contact", None)

        # Check if the emergency_contact table exists
        if emergency_contact_table is None:
            raise KeyError("emergency_contact table does not exist in the database")

        # Initialize an empty list to store matching emergency contacts
        matching_contacts = []

        # Iterate through all emergency contacts in the table
        for contact_id, contact in emergency_contact_table.items():
            # Check if the contact belongs to the specified pet_id
            if contact.pet_id == pet_id:
                # Convert the contact object to a dictionary representation
                contact_dict = {
                    "emergency_contact_id": contact.emergency_contact_id,
                    "pet_id": contact.pet_id,
                    "contact_type": contact.contact_type,
                    "contact_name": contact.contact_name,
                    "phone_number": contact.phone_number,
                    "alternate_phone": contact.alternate_phone,
                    "email": contact.email,
                    "address": contact.address,
                    "available_hours": contact.available_hours,
                    "notes": contact.notes,
                    "creation_timestamp": contact.creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")
                }
                # Add the contact dictionary to the matching contacts list
                matching_contacts.append(contact_dict)

        # Calculate the total count of emergency contacts found
        total_count = len(matching_contacts)

        # Return the result containing the list of emergency contacts and total count
        return {
            "emergency_contacts": matching_contacts,
            "total_count": total_count
        }

    @is_tool()
    def get_exercise_history(self, pet_id: str, start_date: str = None, end_date: str = None):
        """
        Retrieve exercise history for a pet within a date range.

        Args:
            pet_id: Unique identifier of the pet
            start_date: Start date of the range in yyyy-mm-dd format (optional)
            end_date: End date of the range in yyyy-mm-dd format (optional)

        Returns:
            dict: Contains exercise_logs list, total_count, and total_duration_minutes

        Raises:
            KeyError: If exercise_log table not found or pet_id not found
            ValueError: If date format is invalid or date range is invalid
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get exercise_log table from database
        exercise_log_table = getattr(db, 'exercise_log', None)

        # Check if exercise_log table exists
        if exercise_log_table is None:
            raise KeyError("exercise_log table not found in database")

        # Initialize result structure
        exercise_logs = []
        total_count = 0
        total_duration_minutes = 0

        # Parse date range if provided
        start_datetime = None
        end_datetime = None

        if start_date:
            try:
                # Parse start_date string to datetime object (start of day)
                start_datetime = datetime.strptime(start_date, "%Y-%m-%d")
            except ValueError:
                raise ValueError(f"Invalid start_date format: {start_date}. Expected format: yyyy-mm-dd")

        if end_date:
            try:
                # Parse end_date string to datetime object (end of day)
                end_datetime = datetime.strptime(end_date, "%Y-%m-%d")
                # Set to end of day (23:59:59) to include all logs on end_date
                end_datetime = end_datetime.replace(hour=23, minute=59, second=59)
            except ValueError:
                raise ValueError(f"Invalid end_date format: {end_date}. Expected format: yyyy-mm-dd")

        # Validate date range logic
        if start_datetime and end_datetime and start_datetime > end_datetime:
            raise ValueError("start_date cannot be after end_date")

        # Track if we found any logs for this pet_id
        pet_exists = False

        # Iterate through all exercise logs in the table
        for log_id, log_entry in exercise_log_table.items():
            # Filter by pet_id
            if log_entry.pet_id != pet_id:
                continue

            # Mark that we found at least one log for this pet
            pet_exists = True

            # Get the log date (assuming exercise_log has a date/datetime field)
            # Common field names: date, log_date, exercise_date, timestamp, created_at
            log_datetime = None
            for date_field in ['date', 'log_date', 'exercise_date', 'timestamp', 'created_at']:
                if hasattr(log_entry, date_field):
                    log_datetime = getattr(log_entry, date_field)
                    break

            # If no date field found, include the log (no date filtering)
            if log_datetime is None:
                # Build exercise log entry
                log_dict = {
                    'exercise_log_id': log_id
                }
                # Add all available attributes from the log entry
                if hasattr(log_entry, 'exercise_type'):
                    log_dict['exercise_type'] = log_entry.exercise_type
                if hasattr(log_entry, 'duration_minutes'):
                    log_dict['duration_minutes'] = log_entry.duration_minutes
                    total_duration_minutes += log_entry.duration_minutes

                exercise_logs.append(log_dict)
                total_count += 1
                continue

            # Convert log_datetime to datetime if it's a date object
            if isinstance(log_datetime, date) and not isinstance(log_datetime, datetime):
                log_datetime = datetime.combine(log_datetime, datetime.min.time())

            # Apply date range filtering
            if start_datetime and log_datetime < start_datetime:
                continue
            if end_datetime and log_datetime > end_datetime:
                continue

            # Build exercise log entry dictionary
            log_dict = {
                'exercise_log_id': log_id
            }

            # Add all available attributes from the log entry
            if hasattr(log_entry, 'exercise_type'):
                log_dict['exercise_type'] = log_entry.exercise_type
            if hasattr(log_entry, 'duration_minutes'):
                log_dict['duration_minutes'] = log_entry.duration_minutes
                total_duration_minutes += log_entry.duration_minutes

            # Add any other relevant fields that might exist
            for attr in ['distance', 'calories_burned', 'notes', 'location']:
                if hasattr(log_entry, attr):
                    log_dict[attr] = getattr(log_entry, attr)

            exercise_logs.append(log_dict)
            total_count += 1

        # Check if pet exists - if no logs found for this pet_id, raise KeyError
        if not pet_exists:
            raise KeyError(f"No exercise logs found for pet_id: {pet_id}. Pet may not exist in the system.")

        # Return the exercise history
        return {
            'exercise_logs': exercise_logs,
            'total_count': total_count,
            'total_duration_minutes': total_duration_minutes
        }

    @is_tool()
    def create_insurance_claim(
        self,
        pet_id: str = None,
        policy_id: str = None,
        claim_date: str = None,
        incident_date: str = None,
        claim_type: Literal["illness", "injury", "surgery", "emergency", "preventive_care", "dental", "medication", "other"] = None,
        total_cost: float = None,
        diagnosis: str = None,
        treatment_description: str = None,
        veterinarian_name: str = None,
        clinic_id: str = None
    ):
        """
        Create a new pet insurance claim record in the database.

        This method validates input parameters, generates unique identifiers,
        and creates a new insurance claim with initial status 'submitted'.
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters
        if not pet_id or not isinstance(pet_id, str):
            raise ValueError("pet_id must be a non-empty string")

        if not policy_id or not isinstance(policy_id, str):
            raise ValueError("policy_id must be a non-empty string")

        if not claim_date or not isinstance(claim_date, str):
            raise ValueError("claim_date must be a non-empty string in yyyy-mm-dd format")

        if not incident_date or not isinstance(incident_date, str):
            raise ValueError("incident_date must be a non-empty string in yyyy-mm-dd format")

        if not claim_type or claim_type not in ["illness", "injury", "surgery", "emergency", "preventive_care", "dental", "medication", "other"]:
            raise ValueError("claim_type must be one of: illness, injury, surgery, emergency, preventive_care, dental, medication, other")

        if total_cost is None or not isinstance(total_cost, (int, float)) or total_cost < 0:
            raise ValueError("total_cost must be a non-negative number")

        # Parse and validate date formats
        try:
            claim_date_obj = datetime.strptime(claim_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError("claim_date must be in yyyy-mm-dd format")

        try:
            incident_date_obj = datetime.strptime(incident_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError("incident_date must be in yyyy-mm-dd format")

        # Validate that incident_date is not after claim_date
        if incident_date_obj > claim_date_obj:
            raise ValueError("incident_date cannot be after claim_date")

        # Access database
        db = self.db

        # Initialize insurance_claim table if it doesn't exist
        if getattr(db, "insurance_claim", None) is None:
            setattr(db, "insurance_claim", {})

        insurance_claim_table = getattr(db, "insurance_claim")

        # Generate unique claim_id using secure hash
        claim_id = "claim_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Ensure claim_id is unique
        while claim_id in insurance_claim_table:
            claim_id = "claim_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Generate claim_number in format CLM-YYYY-NNNN
        # Count existing claims to generate sequential number
        year = claim_date_obj.year
        claim_count = len([c for c in insurance_claim_table.values() if c.claim_date.year == year])
        claim_number = f"CLM-{year}-{claim_count + 1:04d}"

        # Get current timestamp for submission
        submission_timestamp = datetime.now()

        # Create InsuranceClaim instance
        new_claim = InsuranceClaim(
            claim_id=claim_id,
            pet_id=pet_id,
            policy_id=policy_id,
            claim_number=claim_number,
            claim_date=claim_date_obj,
            incident_date=incident_date_obj,
            claim_type=claim_type,
            diagnosis=diagnosis,
            treatment_description=treatment_description,
            total_cost=float(total_cost),
            veterinarian_name=veterinarian_name,
            clinic_id=clinic_id,
            status="submitted",
            approved_amount=None,
            denial_reason=None,
            submission_timestamp=submission_timestamp,
            last_updated=submission_timestamp
        )

        # Add new claim to database
        insurance_claim_table[claim_id] = new_claim
        setattr(db, "insurance_claim", insurance_claim_table)

        # Calculate estimated processing days (standard 10 business days)
        estimated_processing_days = 10

        # Return claim creation result
        return {
            "claim_id": claim_id,
            "claim_number": claim_number,
            "submission_timestamp": submission_timestamp.strftime("%Y-%m-%d %H:%M:%S"),
            "estimated_processing_days": estimated_processing_days
        }

    @is_tool()
    def get_pet_profile(self, pet_id: str):
        """
        Retrieve complete pet profile information by pet ID

        Args:
            pet_id: Unique identifier of the pet to retrieve

        Returns:
            dict: Complete pet profile information including:
                - pet_id: Unique identifier of the pet
                - pet_name: Name of the pet
                - species: Species of the pet
                - breed: Breed of the pet
                - birth_date: Birth date in yyyy-mm-dd format
                - weight_kg: Current weight in kilograms
                - gender: Gender of the pet
                - microchip_id: Microchip identification number
                - owner_id: Unique identifier of the pet owner

        Raises:
            KeyError: If pet_id does not exist in the system
        """
        # Access the database instance
        db = self.db

        # Retrieve the pet_profile table from the database
        pet_profile_table = getattr(db, "pet_profile", None)

        # Check if the pet_profile table exists
        if pet_profile_table is None:
            raise KeyError(f"Pet profile table not found in database")

        # Check if the pet_id exists in the pet_profile table
        if pet_id not in pet_profile_table:
            raise KeyError(f"Pet with ID '{pet_id}' does not exist in the system")

        # Retrieve the pet profile record
        pet_record = pet_profile_table[pet_id]

        # Convert birth_date from date object to string format yyyy-mm-dd
        birth_date_str = pet_record.birth_date.strftime("%Y-%m-%d")

        # Construct the return dictionary with all required fields
        # Return None for optional fields instead of empty strings or default values
        pet_profile_data = {
            "pet_id": pet_record.pet_id,
            "pet_name": pet_record.pet_name,
            "species": pet_record.species,
            "breed": pet_record.breed,
            "birth_date": birth_date_str,
            "weight_kg": pet_record.weight_kg,
            "gender": pet_record.gender,
            "microchip_id": pet_record.microchip_id,
            "owner_id": pet_record.owner_id
        }

        return pet_profile_data

    @is_tool()
    def get_pet_allergies(self, pet_id: str):
        """
        Retrieve all allergy records for a pet

        This method fetches all allergy records associated with a specific pet from the database.
        It validates that the pet_id exists in the system and returns all matching allergy records.

        Args:
            pet_id: Unique identifier of the pet

        Returns:
            dict containing:
                - allergies: List of all allergy records for the pet
                - total_count: Total number of allergy records found

        Raises:
            KeyError: If pet_id does not exist in the system (no allergy records found)
        """
        # Access the database instance
        db = self.db

        # Get the allergy_record table from the database
        allergy_table = getattr(db, "allergy_record", None)

        # Check if the allergy_record table exists in the database
        if allergy_table is None:
            raise KeyError(f"Allergy record table not found in database")

        # Initialize list to store matching allergy records
        matching_allergies = []

        # Iterate through all allergy records in the table
        for allergy_id, allergy_record in allergy_table.items():
            # Check if the allergy record belongs to the specified pet
            if allergy_record.pet_id == pet_id:
                # Convert the allergy record to a dictionary format
                allergy_dict = {
                    "allergy_id": allergy_record.allergy_id,
                    "pet_id": allergy_record.pet_id,
                    "allergen": allergy_record.allergen,
                    "allergy_type": allergy_record.allergy_type,
                    "severity": allergy_record.severity,
                    "symptoms": allergy_record.symptoms,
                    "diagnosed_date": allergy_record.diagnosed_date.strftime("%Y-%m-%d") if allergy_record.diagnosed_date else None,
                    "veterinarian_name": allergy_record.veterinarian_name,
                    "notes": allergy_record.notes,
                    "creation_timestamp": allergy_record.creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")
                }
                matching_allergies.append(allergy_dict)

        # Check if any allergy records were found for the pet
        # If no records found, raise KeyError as per the pre-condition requirement
        if len(matching_allergies) == 0:
            raise KeyError(f"No allergy records found for pet_id: {pet_id}. Pet profile may not exist in the system.")

        # Return the result with allergies list and total count
        return {
            "allergies": matching_allergies,
            "total_count": len(matching_allergies)
        }

    @is_tool()
    def get_insurance_claim_status(self, claim_id: str):
        """
        Retrieve current status of an insurance claim

        This method retrieves the status and details of an insurance claim by its unique identifier.
        It returns comprehensive information including claim number, status, dates, and financial details.

        Args:
            claim_id: Unique identifier of the insurance claim

        Returns:
            Dictionary containing claim status information with the following keys:
            - claim_id: Unique identifier of the claim
            - claim_number: Claim reference number
            - status: Current status (submitted/under_review/approved/partially_approved/denied/paid)
            - submitted_date: Date when claim was submitted (yyyy-mm-dd format)
            - approved_amount: Approved reimbursement amount (None if not approved)
            - denial_reason: Reason for denial if applicable (None if not denied)
            - last_updated: Last update timestamp (yyyy-mm-dd HH:MM:SS format)

        Raises:
            KeyError: If claim_id does not exist in the database
        """
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Retrieve the insurance_claim table from the database
        insurance_claim_table = getattr(db, 'insurance_claim', None)

        # Check if the insurance_claim table exists
        if insurance_claim_table is None:
            raise KeyError(f"Insurance claim table not found in database")

        # Retrieve the specific claim by claim_id
        # The table stores data as Dict[str, InsuranceClaim] where key is claim_id
        if claim_id not in insurance_claim_table:
            raise KeyError(f"Insurance claim with claim_id '{claim_id}' not found")

        # Get the claim object
        claim = insurance_claim_table[claim_id]

        # Format the submitted_date from date object to yyyy-mm-dd string format
        submitted_date_str = claim.claim_date.strftime("%Y-%m-%d")

        # Format the last_updated from datetime object to yyyy-mm-dd HH:MM:SS string format
        last_updated_str = claim.last_updated.strftime("%Y-%m-%d %H:%M:%S")

        # Construct the return dictionary with all required fields
        result = {
            "claim_id": claim.claim_id,
            "claim_number": claim.claim_number,
            "status": claim.status,
            "submitted_date": submitted_date_str,
            "approved_amount": claim.approved_amount,  # Will be None if not approved
            "denial_reason": claim.denial_reason,  # Will be None if not denied
            "last_updated": last_updated_str
        }

        return result

    @is_tool()
    def calculate_medication_schedule(
        self,
        start_date: str,
        end_date: str,
        frequency: str,
        preferred_times: list = None,
        with_food: bool = False
    ):
        """
        Calculate detailed medication administration schedule based on start/end dates and frequency.

        This method parses the frequency string to determine how many times per day medication
        should be administered, generates a complete schedule across the date range, and
        calculates total doses and duration.

        Args:
            start_date: Start date in yyyy-mm-dd format
            end_date: End date in yyyy-mm-dd format
            frequency: Frequency description (e.g., "once daily", "twice daily", "three times daily")
            preferred_times: Optional list of preferred administration times in HH:MM format
            with_food: Whether medication should be given with food

        Returns:
            dict containing:
                - schedule: List of administration events with date, time, and with_food flag
                - total_doses: Total number of doses in the schedule
                - duration_days: Total duration in days

        Raises:
            ValueError: If date format is invalid, end_date is before start_date, 
                       frequency cannot be parsed, or preferred_times don't match frequency
        """
        from datetime import datetime, timedelta

        # Parse and validate dates
        try:
            start_dt = datetime.strptime(start_date, "%Y-%m-%d").date()
            end_dt = datetime.strptime(end_date, "%Y-%m-%d").date()
        except ValueError as e:
            raise ValueError(f"Invalid date format. Expected yyyy-mm-dd format. Error: {str(e)}")

        # Validate date range
        if end_dt < start_dt:
            raise ValueError("End date cannot be before start date")

        # Calculate duration in days (inclusive of both start and end dates)
        duration_days = (end_dt - start_dt).days + 1

        # Parse frequency to determine doses per day
        # Common patterns: "once daily", "twice daily", "three times daily", "every 8 hours", etc.
        frequency_lower = frequency.lower().strip()

        # Map common frequency patterns to doses per day
        doses_per_day = None

        # Check for numeric patterns (e.g., "1 time daily", "2 times daily")
        if "once" in frequency_lower or "1 time" in frequency_lower or "single" in frequency_lower:
            doses_per_day = 1
        elif "twice" in frequency_lower or "2 times" in frequency_lower or "two times" in frequency_lower:
            doses_per_day = 2
        elif "three times" in frequency_lower or "3 times" in frequency_lower or "thrice" in frequency_lower:
            doses_per_day = 3
        elif "four times" in frequency_lower or "4 times" in frequency_lower:
            doses_per_day = 4
        elif "five times" in frequency_lower or "5 times" in frequency_lower:
            doses_per_day = 5
        elif "six times" in frequency_lower or "6 times" in frequency_lower:
            doses_per_day = 6

        # Check for "every X hours" pattern
        if doses_per_day is None:
            import re
            hours_match = re.search(r'every\s+(\d+)\s+hours?', frequency_lower)
            if hours_match:
                hours = int(hours_match.group(1))
                if hours > 0 and 24 % hours == 0:
                    doses_per_day = 24 // hours
                else:
                    # For non-divisible hours, calculate approximate doses per day
                    doses_per_day = max(1, round(24 / hours))

        # If still not determined, try to extract any number from frequency
        if doses_per_day is None:
            import re
            number_match = re.search(r'\b(\d+)\b', frequency_lower)
            if number_match:
                doses_per_day = int(number_match.group(1))
            else:
                raise ValueError(f"Cannot parse frequency: '{frequency}'. Please specify doses per day clearly.")

        # Validate doses_per_day
        if doses_per_day <= 0:
            raise ValueError(f"Invalid frequency resulting in {doses_per_day} doses per day")

        # Determine administration times
        if preferred_times:
            # Validate preferred_times format and count
            if len(preferred_times) != doses_per_day:
                raise ValueError(
                    f"Number of preferred times ({len(preferred_times)}) does not match "
                    f"frequency ({doses_per_day} doses per day)"
                )

            # Validate time format
            admin_times = []
            for time_str in preferred_times:
                try:
                    # Validate HH:MM format
                    datetime.strptime(time_str, "%H:%M")
                    admin_times.append(time_str)
                except ValueError:
                    raise ValueError(f"Invalid time format: '{time_str}'. Expected HH:MM format.")
        else:
            # Generate evenly spaced times throughout the day
            # Start at 08:00 by default and space evenly
            admin_times = []
            if doses_per_day == 1:
                admin_times = ["08:00"]
            elif doses_per_day == 2:
                admin_times = ["08:00", "20:00"]
            elif doses_per_day == 3:
                admin_times = ["08:00", "14:00", "20:00"]
            elif doses_per_day == 4:
                admin_times = ["08:00", "12:00", "16:00", "20:00"]
            else:
                # For other frequencies, space evenly across 24 hours
                interval_hours = 24 / doses_per_day
                start_hour = 8  # Start at 8 AM
                for i in range(doses_per_day):
                    hour = (start_hour + int(i * interval_hours)) % 24
                    admin_times.append(f"{hour:02d}:00")

        # Generate schedule
        schedule = []
        current_date = start_dt

        while current_date <= end_dt:
            # Add all doses for this day
            for time_str in admin_times:
                schedule.append({
                    "date": current_date.strftime("%Y-%m-%d"),
                    "time": time_str,
                    "with_food": with_food
                })

            # Move to next day
            current_date += timedelta(days=1)

        # Calculate total doses
        total_doses = len(schedule)

        return {
            "schedule": schedule,
            "total_doses": total_doses,
            "duration_days": duration_days
        }

    @is_tool()
    def check_vaccination_due(self, pet_id: str, reference_date: str = None, advance_notice_days: int = 30):
        # Import necessary modules for date handling
        from datetime import datetime, timedelta

        # Get the database instance
        db = self.db

        # Retrieve the vaccination_record table from the database
        vaccination_table = getattr(db, "vaccination_record", None)

        # Raise KeyError if the vaccination_record table doesn't exist
        if vaccination_table is None:
            raise KeyError("vaccination_record table not found in database")

        # Parse the reference_date or use current date if not provided
        if reference_date:
            try:
                # Parse the reference date in yyyy-mm-dd format
                ref_date = datetime.strptime(reference_date, "%Y-%m-%d").date()
            except ValueError:
                raise ValueError(f"Invalid reference_date format: {reference_date}. Expected yyyy-mm-dd format")
        else:
            # Use current date if reference_date is not provided
            ref_date = datetime.now().date()

        # Calculate the future date threshold for upcoming vaccinations
        # Vaccinations due within advance_notice_days from reference_date are considered upcoming
        future_threshold = ref_date + timedelta(days=advance_notice_days)

        # Initialize lists to store overdue and upcoming vaccination records
        overdue_vaccinations = []
        upcoming_vaccinations = []

        # Iterate through all vaccination records in the table
        for vaccination_id, vaccination_record in vaccination_table.items():
            # Check if this vaccination record belongs to the specified pet
            if vaccination_record.pet_id != pet_id:
                continue

            # Get the next due date for this vaccination
            next_due = vaccination_record.next_due_date

            # Check if the vaccination is overdue (due date is before reference date)
            if next_due < ref_date:
                overdue_vaccinations.append({
                    "vaccination_id": vaccination_record.vaccination_id,
                    "vaccine_name": vaccination_record.vaccine_name,
                    "due_date": next_due.strftime("%Y-%m-%d")
                })
            # Check if the vaccination is upcoming (due date is between reference date and future threshold)
            elif ref_date <= next_due <= future_threshold:
                upcoming_vaccinations.append({
                    "vaccination_id": vaccination_record.vaccination_id,
                    "vaccine_name": vaccination_record.vaccine_name,
                    "due_date": next_due.strftime("%Y-%m-%d")
                })

        # Sort overdue vaccinations by due date (earliest first)
        overdue_vaccinations.sort(key=lambda x: x["due_date"])

        # Sort upcoming vaccinations by due date (earliest first)
        upcoming_vaccinations.sort(key=lambda x: x["due_date"])

        # Determine if there are any overdue vaccinations
        has_overdue = len(overdue_vaccinations) > 0

        # Return the result containing overdue, upcoming vaccinations and overdue status
        return {
            "overdue_vaccinations": overdue_vaccinations,
            "upcoming_vaccinations": upcoming_vaccinations,
            "has_overdue": has_overdue
        }

    @is_tool()
    def book_service_appointment(
        self,
        pet_id: str,
        service_type: Literal["grooming", "veterinary_checkup", "vaccination", "dental_care", "surgery", "emergency", "boarding", "training", "behavioral_consultation"],
        appointment_datetime: str,
        provider_id: str,
        duration_minutes: int = None,
        location_id: str = None,
        special_requests: str = None
    ):
        """
        Book a service appointment for a pet.

        This method creates a new appointment record in the database with the provided details.
        It validates the input parameters, generates unique identifiers, and stores the appointment.
        """
        # Import required modules for ID generation and datetime handling
        import secrets
        import hashlib
        from datetime import datetime

        # Validate required parameters are not empty
        if not pet_id or not isinstance(pet_id, str) or not pet_id.strip():
            raise ValueError("pet_id must be a non-empty string")

        if not provider_id or not isinstance(provider_id, str) or not provider_id.strip():
            raise ValueError("provider_id must be a non-empty string")

        # Validate service_type is within allowed enum values (additional safety check)
        allowed_service_types = [
            "grooming", "veterinary_checkup", "vaccination", "dental_care",
            "surgery", "emergency", "boarding", "training", "behavioral_consultation"
        ]
        if service_type not in allowed_service_types:
            raise ValueError(f"service_type must be one of {allowed_service_types}, got '{service_type}'")

        # Validate and parse appointment_datetime format
        if not appointment_datetime or not isinstance(appointment_datetime, str):
            raise ValueError("appointment_datetime must be a non-empty string")

        try:
            # Parse the datetime string to ensure it's in the correct format
            parsed_datetime = datetime.strptime(appointment_datetime, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"appointment_datetime must be in 'yyyy-mm-dd HH:MM:SS' format, got '{appointment_datetime}': {str(e)}")

        # Validate duration_minutes if provided
        if duration_minutes is not None:
            if not isinstance(duration_minutes, int) or duration_minutes <= 0:
                raise ValueError("duration_minutes must be a positive integer")

        # Validate location_id if provided
        if location_id is not None and (not isinstance(location_id, str) or not location_id.strip()):
            raise ValueError("location_id must be a non-empty string if provided")

        # Validate special_requests if provided
        if special_requests is not None and not isinstance(special_requests, str):
            raise ValueError("special_requests must be a string if provided")

        # Access the database
        db = self.db

        # Initialize appointment table if it doesn't exist
        if getattr(db, "appointment", None) is None:
            setattr(db, "appointment", {})

        # Get the appointment table
        appointment_table = getattr(db, "appointment")

        # Generate unique appointment_id
        appointment_id = "appt_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Ensure the generated ID is unique
        while appointment_id in appointment_table:
            appointment_id = "appt_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Generate unique confirmation_code
        confirmation_code = "CONF-" + datetime.now().strftime("%Y") + "-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:8].upper()

        # Get current timestamp for booking_timestamp
        booking_timestamp = datetime.now()

        # Create the appointment object
        new_appointment = Appointment(
            appointment_id=appointment_id,
            pet_id=pet_id,
            service_type=service_type,
            appointment_datetime=parsed_datetime,
            duration_minutes=duration_minutes,
            provider_id=provider_id,
            location_id=location_id,
            special_requests=special_requests,
            confirmation_code=confirmation_code,
            status="scheduled",  # Default status for new appointments
            booking_timestamp=booking_timestamp
        )

        # Add the appointment to the database
        appointment_table[appointment_id] = new_appointment
        setattr(db, "appointment", appointment_table)

        # Return the required information
        return {
            "appointment_id": appointment_id,
            "confirmation_code": confirmation_code,
            "booking_timestamp": booking_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def get_upcoming_appointments(self, pet_id: str, from_date: str = None, days_ahead: int = 30):
        """
        Retrieve upcoming appointments for a pet within a specified date range.

        Args:
            pet_id: Unique identifier of the pet
            from_date: Start date in yyyy-mm-dd format (defaults to current date if not provided)
            days_ahead: Number of days ahead to retrieve appointments (default: 30)

        Returns:
            dict: Contains 'appointments' list and 'total_count' of upcoming appointments

        Raises:
            KeyError: If pet_id does not exist in any appointment records
        """
        from datetime import datetime, timedelta

        # Access the database
        db = self.db

        # Get the appointment table
        appointment_table = getattr(db, "appointment", None)

        # If appointment table doesn't exist or is empty, raise KeyError
        if appointment_table is None or len(appointment_table) == 0:
            raise KeyError(f"No appointments found for pet_id: {pet_id}")

        # Parse from_date or use current date
        if from_date:
            try:
                start_date = datetime.strptime(from_date, "%Y-%m-%d")
            except ValueError:
                raise ValueError(f"Invalid from_date format. Expected yyyy-mm-dd, got: {from_date}")
        else:
            # Use current date (set time to start of day for comparison)
            start_date = datetime.now().replace(hour=0, minute=0, second=0, microsecond=0)

        # Calculate end date based on days_ahead
        end_date = start_date + timedelta(days=days_ahead)

        # Filter appointments for the specified pet_id and date range
        upcoming_appointments = []
        pet_has_appointments = False

        for appointment_id, appointment in appointment_table.items():
            # Check if this appointment belongs to the specified pet
            if appointment.pet_id == pet_id:
                pet_has_appointments = True

                # Check if appointment is within the date range and not cancelled
                appointment_dt = appointment.appointment_datetime

                # Only include appointments that are:
                # 1. Within the date range (>= start_date and < end_date)
                # 2. Not cancelled (status != "cancelled")
                if start_date <= appointment_dt < end_date and appointment.status != "cancelled":
                    # Format appointment data for return
                    appointment_data = {
                        "appointment_id": appointment.appointment_id,
                        "service_type": appointment.service_type,
                        "appointment_datetime": appointment_dt.strftime("%Y-%m-%d %H:%M:%S"),
                        "duration_minutes": appointment.duration_minutes,
                        "provider_id": appointment.provider_id,
                        "location_id": appointment.location_id,
                        "special_requests": appointment.special_requests,
                        "confirmation_code": appointment.confirmation_code,
                        "status": appointment.status,
                        "booking_timestamp": appointment.booking_timestamp.strftime("%Y-%m-%d %H:%M:%S")
                    }
                    upcoming_appointments.append(appointment_data)

        # If no appointments found for this pet_id at all, raise KeyError
        if not pet_has_appointments:
            raise KeyError(f"No appointments found for pet_id: {pet_id}")

        # Sort appointments by appointment_datetime (earliest first)
        upcoming_appointments.sort(key=lambda x: x["appointment_datetime"])

        # Return the result
        return {
            "appointments": upcoming_appointments,
            "total_count": len(upcoming_appointments)
        }

    @is_tool()
    def calculate_exercise_statistics(self, pet_id: str, start_date: str, end_date: str):
        """
        Calculate exercise statistics for a pet over a specified period.

        Args:
            pet_id: Unique identifier of the pet
            start_date: Start date of analysis period in yyyy-mm-dd format
            end_date: End date of analysis period in yyyy-mm-dd format

        Returns:
            Dictionary containing exercise statistics including total days, average duration,
            total distance, most common activity, total calories, and exercise frequency

        Raises:
            KeyError: If pet_id not found in exercise logs or database access fails
            ValueError: If date format is invalid or date range is invalid
        """
        from datetime import datetime
        from collections import Counter

        # Access the database
        db = self.db

        # Get exercise_log table from database
        exercise_log_table = getattr(db, 'exercise_log', None)
        if exercise_log_table is None:
            raise KeyError("exercise_log table not found in database")

        # Validate and parse date strings
        try:
            start_dt = datetime.strptime(start_date, "%Y-%m-%d")
            end_dt = datetime.strptime(end_date, "%Y-%m-%d")
        except ValueError as e:
            raise ValueError(f"Invalid date format. Expected yyyy-mm-dd format: {str(e)}")

        # Validate date range
        if start_dt > end_dt:
            raise ValueError("start_date cannot be after end_date")

        # Filter exercise logs for the specified pet and date range
        pet_exercises = []
        pet_found = False

        for log_id, exercise_log in exercise_log_table.items():
            # Check if this log belongs to the specified pet
            if exercise_log.pet_id == pet_id:
                pet_found = True

                # Parse the exercise date from the log
                # Assuming exercise_log has a date field (datetime object or string)
                if hasattr(exercise_log, 'date'):
                    log_date = exercise_log.date
                    if isinstance(log_date, str):
                        # Parse string date - handle both date-only and datetime formats
                        try:
                            # Try parsing as datetime first
                            if ' ' in log_date:
                                log_date = datetime.strptime(log_date, "%Y-%m-%d %H:%M:%S")
                            else:
                                log_date = datetime.strptime(log_date, "%Y-%m-%d")
                        except ValueError:
                            continue
                    elif isinstance(log_date, datetime):
                        # Already datetime object, use as is
                        pass
                    else:
                        continue

                    # Check if log date is within the specified range (compare dates only)
                    log_date_only = log_date.replace(hour=0, minute=0, second=0, microsecond=0)
                    if start_dt <= log_date_only <= end_dt:
                        pet_exercises.append(exercise_log)

        # Raise KeyError if pet not found in any exercise logs
        if not pet_found:
            raise KeyError(f"Pet with id '{pet_id}' not found in exercise logs")

        # If no exercises found in the date range, return zero statistics
        if not pet_exercises:
            return {
                "total_exercise_days": 0,
                "average_duration_minutes": 0.0,
                "total_distance_km": 0.0,
                "most_common_activity": "",
                "total_calories_burned": 0.0,
                "exercise_frequency_per_week": 0.0
            }

        # Calculate statistics
        # 1. Total exercise days - count unique dates
        exercise_dates = set()
        for exercise in pet_exercises:
            if hasattr(exercise, 'date'):
                log_date = exercise.date
                if isinstance(log_date, str):
                    # Extract date portion
                    if ' ' in log_date:
                        date_str = log_date.split()[0]
                    else:
                        date_str = log_date
                elif isinstance(log_date, datetime):
                    date_str = log_date.strftime("%Y-%m-%d")
                else:
                    continue
                exercise_dates.add(date_str)

        total_exercise_days = len(exercise_dates)

        # 2. Average duration per session
        total_duration = 0.0
        duration_count = 0
        for exercise in pet_exercises:
            if hasattr(exercise, 'duration_minutes'):
                duration = exercise.duration_minutes
                if duration is not None:
                    total_duration += float(duration)
                    duration_count += 1

        average_duration_minutes = round(total_duration / duration_count, 1) if duration_count > 0 else 0.0

        # 3. Total distance covered
        total_distance_km = 0.0
        for exercise in pet_exercises:
            if hasattr(exercise, 'distance_km'):
                distance = exercise.distance_km
                if distance is not None:
                    total_distance_km += float(distance)

        total_distance_km = round(total_distance_km, 1)

        # 4. Most common activity type
        activity_types = []
        for exercise in pet_exercises:
            if hasattr(exercise, 'activity_type'):
                activity = exercise.activity_type
                if activity:
                    activity_types.append(activity)

        if activity_types:
            activity_counter = Counter(activity_types)
            most_common_activity = activity_counter.most_common(1)[0][0]
        else:
            most_common_activity = ""

        # 5. Total calories burned
        total_calories_burned = 0.0
        for exercise in pet_exercises:
            if hasattr(exercise, 'calories_burned'):
                calories = exercise.calories_burned
                if calories is not None:
                    total_calories_burned += float(calories)

        total_calories_burned = round(total_calories_burned, 1)

        # 6. Exercise frequency per week
        # Calculate total weeks in the date range
        date_range_days = (end_dt - start_dt).days + 1
        total_weeks = date_range_days / 7.0

        # Calculate average sessions per week
        total_sessions = len(pet_exercises)
        exercise_frequency_per_week = round(total_sessions / total_weeks, 1) if total_weeks > 0 else 0.0

        # Return statistics
        return {
            "total_exercise_days": total_exercise_days,
            "average_duration_minutes": average_duration_minutes,
            "total_distance_km": total_distance_km,
            "most_common_activity": most_common_activity,
            "total_calories_burned": total_calories_burned,
            "exercise_frequency_per_week": exercise_frequency_per_week
        }

    @is_tool()
    def track_weight_trend(self, pet_id: str, start_date: str, end_date: str):
        # Parse date strings to date objects for comparison
        # Expected format: "yyyy-mm-dd"
        from datetime import datetime, date

        try:
            start_date_obj = datetime.strptime(start_date, "%Y-%m-%d").date()
            end_date_obj = datetime.strptime(end_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError("Invalid date format. Expected format: yyyy-mm-dd")

        # Validate date range
        if start_date_obj > end_date_obj:
            raise ValueError("start_date must be before or equal to end_date")

        # Access database
        db = self.db

        # Get pet profile to verify pet exists
        pet_profile_table = getattr(db, "pet_profile", None)
        if pet_profile_table is None:
            raise KeyError("pet_profile table not found in database")

        pet = pet_profile_table.get(pet_id)
        if pet is None:
            raise KeyError(f"Pet with id '{pet_id}' not found")

        # Get weight history table
        weight_history_table = getattr(db, "weight_history", None)
        if weight_history_table is None:
            raise KeyError("weight_history table not found in database")

        # Filter weight measurements within the date range for this pet
        weight_measurements = []
        for record_id, record in weight_history_table.items():
            # Check if record belongs to this pet
            if record.pet_id == pet_id:
                # Check if measurement date is within range
                if start_date_obj <= record.measurement_date <= end_date_obj:
                    weight_measurements.append({
                        "date": record.measurement_date.strftime("%Y-%m-%d"),
                        "weight_kg": record.weight_kg
                    })

        # Check if there are any measurements
        if not weight_measurements:
            raise KeyError(f"No weight measurements found for pet '{pet_id}' between {start_date} and {end_date}")

        # Sort measurements by date
        weight_measurements.sort(key=lambda x: x["date"])

        # Calculate statistics
        weights = [m["weight_kg"] for m in weight_measurements]
        average_weight_kg = sum(weights) / len(weights)

        # Calculate weight change (last - first)
        first_weight = weights[0]
        last_weight = weights[-1]
        weight_change_kg = last_weight - first_weight

        # Calculate percentage change (avoid division by zero)
        if first_weight > 0:
            weight_change_percentage = (weight_change_kg / first_weight) * 100
        else:
            weight_change_percentage = 0.0

        # Determine trend based on weight change
        # Use a threshold to determine if weight is stable (within ±2% is considered stable)
        if abs(weight_change_percentage) < 2.0:
            trend = "stable"
        elif weight_change_kg > 0:
            trend = "increasing"
        else:
            trend = "decreasing"

        # Generate health assessment
        # Consider both trend and magnitude of change
        if trend == "stable":
            health_assessment = "Weight is stable and within healthy range"
        elif trend == "increasing":
            if weight_change_percentage > 10:
                health_assessment = "Significant weight gain detected. Consider consulting a veterinarian"
            elif weight_change_percentage > 5:
                health_assessment = "Moderate weight gain observed. Monitor diet and exercise"
            else:
                health_assessment = "Slight weight increase noted. Continue monitoring"
        else:  # decreasing
            if weight_change_percentage < -10:
                health_assessment = "Significant weight loss detected. Veterinary consultation recommended"
            elif weight_change_percentage < -5:
                health_assessment = "Moderate weight loss observed. Check diet and health status"
            else:
                health_assessment = "Slight weight decrease noted. Continue monitoring"

        # Round numerical values for cleaner output
        average_weight_kg = round(average_weight_kg, 1)
        weight_change_kg = round(weight_change_kg, 1)
        weight_change_percentage = round(weight_change_percentage, 1)

        # Return analysis results
        return {
            "weight_measurements": weight_measurements,
            "trend": trend,
            "average_weight_kg": average_weight_kg,
            "weight_change_kg": weight_change_kg,
            "weight_change_percentage": weight_change_percentage,
            "health_assessment": health_assessment
        }

    @is_tool()
    def create_behavior_log(
        self,
        pet_id: str,
        observation_datetime: str,
        behavior_type: Literal["aggression", "anxiety", "excessive_barking", "destructive", "separation_anxiety", "fear", "hyperactivity", "lethargy", "appetite_change", "positive", "other"],
        severity: Literal["mild", "moderate", "severe"],
        triggers: Optional[List[str]] = None,
        duration_minutes: Optional[int] = None,
        context: Optional[str] = None,
        intervention_taken: Optional[str] = None
    ) -> dict:
        from datetime import datetime
        import secrets
        import hashlib
        import json

        # Validate required parameters
        if not pet_id or not isinstance(pet_id, str):
            raise ValueError("pet_id must be a non-empty string")

        if not observation_datetime or not isinstance(observation_datetime, str):
            raise ValueError("observation_datetime must be a non-empty string")

        if not behavior_type or not isinstance(behavior_type, str):
            raise ValueError("behavior_type must be a non-empty string")

        if not severity or not isinstance(severity, str):
            raise ValueError("severity must be a non-empty string")

        # Validate and parse observation_datetime format
        try:
            parsed_observation_datetime = datetime.strptime(observation_datetime, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError("observation_datetime must be in 'yyyy-mm-dd HH:MM:SS' format (e.g., '2024-01-15 16:30:00')")

        # Validate behavior_type enum
        valid_behavior_types = [
            "aggression", "anxiety", "excessive_barking", "destructive", 
            "separation_anxiety", "fear", "hyperactivity", "lethargy", 
            "appetite_change", "positive", "other"
        ]
        if behavior_type not in valid_behavior_types:
            raise ValueError(f"behavior_type must be one of {valid_behavior_types}, got '{behavior_type}'")

        # Validate severity enum
        valid_severities = ["mild", "moderate", "severe"]
        if severity not in valid_severities:
            raise ValueError(f"severity must be one of {valid_severities}, got '{severity}'")

        # Validate optional parameters
        if duration_minutes is not None:
            if not isinstance(duration_minutes, int) or duration_minutes < 0:
                raise ValueError("duration_minutes must be a non-negative integer")

        if triggers is not None:
            if not isinstance(triggers, list):
                raise ValueError("triggers must be a list of strings")
            for trigger in triggers:
                if not isinstance(trigger, str):
                    raise ValueError("All triggers must be strings")

        # Generate unique behavior log ID
        prefix = "behlog_"
        behavior_log_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for creation
        creation_timestamp = datetime.now()

        # Convert triggers list to JSON string for storage (if provided)
        triggers_str = None
        if triggers is not None:
            triggers_str = json.dumps(triggers)

        # Access database
        db = self.db

        # Get behavior_log table from database
        behavior_log_table = getattr(db, "behavior_log", None)
        if behavior_log_table is None:
            # Initialize empty table if it doesn't exist
            behavior_log_table = {}

        # Import BehaviorLog class using absolute import (as required)

        # Create new BehaviorLog instance
        new_behavior_log = BehaviorLog(
            behavior_log_id=behavior_log_id,
            pet_id=pet_id,
            observation_datetime=parsed_observation_datetime,
            behavior_type=behavior_type,
            severity=severity,
            triggers=triggers_str,
            duration_minutes=duration_minutes,
            context=context,
            intervention_taken=intervention_taken,
            creation_timestamp=creation_timestamp
        )

        # Add new log entry to the table
        behavior_log_table[behavior_log_id] = new_behavior_log

        # Update database with new table data
        setattr(db, "behavior_log", behavior_log_table)

        # Return result with formatted timestamps
        return {
            "behavior_log_id": behavior_log_id,
            "creation_timestamp": creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def calculate_insurance_coverage(
        self,
        total_cost: float,
        deductible: float,
        coverage_percentage: float,
        annual_limit: float,
        already_claimed_this_year: float = 0.0
    ):
        """
        Calculate insurance coverage amount for a given treatment cost.

        This method computes:
        1. The eligible amount after applying the deductible
        2. The coverage amount based on the policy's coverage percentage
        3. The out-of-pocket amount the pet owner needs to pay
        4. The remaining annual coverage limit after this claim

        Args:
            total_cost: Total cost of treatment (must be non-negative)
            deductible: Policy deductible amount (must be non-negative)
            coverage_percentage: Coverage percentage of the policy (0-100)
            annual_limit: Annual coverage limit (must be positive)
            already_claimed_this_year: Amount already claimed in current year (default: 0.0)

        Returns:
            Dictionary containing:
            - eligible_amount: Amount eligible for coverage after deductible
            - coverage_amount: Calculated coverage amount
            - out_of_pocket: Amount pet owner needs to pay
            - remaining_annual_limit: Remaining coverage available this year
            - coverage_percentage_applied: Coverage percentage applied

        Raises:
            ValueError: If any input parameters are invalid
        """
        # Input validation
        if total_cost < 0:
            raise ValueError("total_cost must be non-negative")

        if deductible < 0:
            raise ValueError("deductible must be non-negative")

        if coverage_percentage < 0 or coverage_percentage > 100:
            raise ValueError("coverage_percentage must be between 0 and 100")

        if annual_limit <= 0:
            raise ValueError("annual_limit must be positive")

        if already_claimed_this_year < 0:
            raise ValueError("already_claimed_this_year must be non-negative")

        # Check if annual limit has been exceeded
        if already_claimed_this_year >= annual_limit:
            raise ValueError("Annual limit has already been reached or exceeded")

        # Step 1: Calculate eligible amount after deductible
        # The deductible is subtracted from the total cost
        # If total cost is less than deductible, eligible amount is 0
        eligible_amount = max(0.0, total_cost - deductible)

        # Step 2: Calculate coverage amount based on coverage percentage
        # Convert percentage to decimal (e.g., 80% -> 0.80)
        coverage_decimal = coverage_percentage / 100.0
        calculated_coverage = eligible_amount * coverage_decimal

        # Step 3: Apply annual limit constraint
        # Calculate remaining limit available for this claim
        remaining_limit = annual_limit - already_claimed_this_year

        # The actual coverage amount cannot exceed the remaining annual limit
        coverage_amount = min(calculated_coverage, remaining_limit)

        # Step 4: Calculate out-of-pocket amount
        # This is the total cost minus the coverage amount
        out_of_pocket = total_cost - coverage_amount

        # Step 5: Calculate remaining annual limit after this claim
        remaining_annual_limit = annual_limit - already_claimed_this_year - coverage_amount

        # Ensure remaining limit doesn't go negative due to floating point precision
        remaining_annual_limit = max(0.0, remaining_annual_limit)

        # Return the calculation results
        return {
            "eligible_amount": round(eligible_amount, 2),
            "coverage_amount": round(coverage_amount, 2),
            "out_of_pocket": round(out_of_pocket, 2),
            "remaining_annual_limit": round(remaining_annual_limit, 2),
            "coverage_percentage_applied": coverage_percentage
        }

    @is_tool()
    def create_training_session(
        self,
        pet_id: str,
        session_date: str,
        training_type: Literal["obedience", "agility", "behavior_modification", "socialization", "trick_training", "potty_training", "leash_training", "other"],
        duration_minutes: int,
        commands_practiced: List[str] = None,
        success_rate: float = None,
        trainer_name: str = None,
        notes: str = None
    ) -> dict:
        """
        Create a training session record for a pet.

        This method creates a new training session record with the provided information,
        generates a unique training_session_id, and stores it in the database.

        Args:
            pet_id: Unique identifier of the pet
            session_date: Date of the training session in yyyy-mm-dd format
            training_type: Type of training conducted (must be one of the enum values)
            duration_minutes: Duration of the training session in minutes
            commands_practiced: Optional list of commands or skills practiced
            success_rate: Optional success rate of command execution as percentage
            trainer_name: Optional name of the trainer conducting the session
            notes: Optional additional notes about the training session

        Returns:
            Dictionary containing:
            - training_session_id: Unique identifier for the training session record
            - creation_timestamp: Timestamp when the record was created in yyyy-mm-dd HH:MM:SS format

        Raises:
            ValueError: If required parameters are invalid or missing
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Validate required parameters
        if not pet_id or not isinstance(pet_id, str):
            raise ValueError("pet_id must be a non-empty string")

        if not session_date or not isinstance(session_date, str):
            raise ValueError("session_date must be a non-empty string")

        if not isinstance(duration_minutes, int) or duration_minutes <= 0:
            raise ValueError("duration_minutes must be a positive integer")

        # Validate and parse session_date format (yyyy-mm-dd)
        try:
            parsed_date = datetime.strptime(session_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError("session_date must be in yyyy-mm-dd format")

        # Validate training_type is one of the allowed enum values
        allowed_training_types = [
            "obedience", "agility", "behavior_modification", "socialization",
            "trick_training", "potty_training", "leash_training", "other"
        ]
        if training_type not in allowed_training_types:
            raise ValueError(f"training_type must be one of {allowed_training_types}")

        # Validate optional parameters if provided
        if success_rate is not None:
            if not isinstance(success_rate, (int, float)) or success_rate < 0 or success_rate > 100:
                raise ValueError("success_rate must be a number between 0 and 100")

        if commands_practiced is not None:
            if not isinstance(commands_practiced, list):
                raise ValueError("commands_practiced must be a list of strings")
            if not all(isinstance(cmd, str) for cmd in commands_practiced):
                raise ValueError("All items in commands_practiced must be strings")

        if trainer_name is not None and not isinstance(trainer_name, str):
            raise ValueError("trainer_name must be a string")

        if notes is not None and not isinstance(notes, str):
            raise ValueError("notes must be a string")

        # Access the database
        db = self.db

        # Generate unique training_session_id
        prefix = "train_"
        training_session_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create current timestamp
        creation_timestamp = datetime.now()

        # Convert commands_practiced list to JSON string for storage if provided
        commands_practiced_str = None
        if commands_practiced is not None:
            import json
            commands_practiced_str = json.dumps(commands_practiced)

        # Import the TrainingSession class (already imported at file header)
        from datetime import date

        # Create new training session record
        new_training_session = TrainingSession(
            training_session_id=training_session_id,
            pet_id=pet_id,
            session_date=parsed_date,
            training_type=training_type,
            duration_minutes=duration_minutes,
            commands_practiced=commands_practiced_str,
            success_rate=success_rate,
            trainer_name=trainer_name,
            notes=notes,
            creation_timestamp=creation_timestamp
        )

        # Get existing training_session table or initialize empty dict
        training_session_table = getattr(db, "training_session", None)
        if training_session_table is None:
            training_session_table = {}

        # Add new training session to the table
        training_session_table[training_session_id] = new_training_session

        # Save updated table back to database
        setattr(db, "training_session", training_session_table)

        # Return the training_session_id and creation_timestamp in the required format
        return {
            "training_session_id": training_session_id,
            "creation_timestamp": creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def create_multi_pet_household(self, household_name: str, owner_id: str, pet_ids: List[str], address: str = None, shared_veterinarian_id: str = None):
        """
        Create a household profile for managing multiple pets.

        This method creates a new household entry and associates multiple pets with it.
        It generates a unique household_id, stores the household information in the database,
        and creates household-pet associations for each pet in the pet_ids list.

        Args:
            household_name: Name for the household (e.g., "Smith Family Pets")
            owner_id: Unique identifier of the primary owner
            pet_ids: List of pet identifiers to be added to the household
            address: Optional household address
            shared_veterinarian_id: Optional shared veterinarian for all pets

        Returns:
            dict: Contains household_id, creation_timestamp, and total_pets

        Raises:
            ValueError: If household_name is empty, pet_ids list is empty, or any required field is invalid
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Validate required parameters
        if not household_name or not household_name.strip():
            raise ValueError("household_name cannot be empty")

        if not owner_id or not owner_id.strip():
            raise ValueError("owner_id cannot be empty")

        if not pet_ids or len(pet_ids) == 0:
            raise ValueError("pet_ids list cannot be empty")

        # Validate that all pet_ids are non-empty strings
        for pet_id in pet_ids:
            if not pet_id or not isinstance(pet_id, str) or not pet_id.strip():
                raise ValueError(f"Invalid pet_id in pet_ids list: {pet_id}")

        # Access the database
        db = self.db

        # Generate unique household_id
        prefix = "household_"
        household_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        creation_timestamp = datetime.now()

        # Create Household object
        new_household = Household(
            household_id=household_id,
            household_name=household_name.strip(),
            owner_id=owner_id.strip(),
            address=address.strip() if address else None,
            shared_veterinarian_id=shared_veterinarian_id.strip() if shared_veterinarian_id else None,
            creation_timestamp=creation_timestamp
        )

        # Get existing household data or initialize empty dict
        household_data = getattr(db, "household", None)
        if household_data is None:
            household_data = {}

        # Add new household to database
        household_data[household_id] = new_household
        setattr(db, "household", household_data)

        # Get existing household_pet data or initialize empty dict
        household_pet_data = getattr(db, "household_pet", None)
        if household_pet_data is None:
            household_pet_data = {}

        # Create household-pet associations for each pet
        added_timestamp = datetime.now()
        for pet_id in pet_ids:
            # Generate unique household_pet_id
            household_pet_id = "hp_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Create HouseholdPet object
            household_pet = HouseholdPet(
                household_pet_id=household_pet_id,
                household_id=household_id,
                pet_id=pet_id.strip(),
                added_timestamp=added_timestamp
            )

            # Add to household_pet data
            household_pet_data[household_pet_id] = household_pet

        # Save household_pet data back to database
        setattr(db, "household_pet", household_pet_data)

        # Calculate total number of pets
        total_pets = len(pet_ids)

        # Format creation_timestamp as string in "yyyy-mm-dd HH:MM:SS" format
        creation_timestamp_str = creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return household creation information
        return {
            "household_id": household_id,
            "creation_timestamp": creation_timestamp_str,
            "total_pets": total_pets
        }

    @is_tool()
    def create_wellness_reminder(
        self,
        pet_id: str,
        reminder_type: Literal["vaccination", "medication", "checkup", "grooming", "flea_treatment", "deworming", "dental_care", "exercise", "diet_review"],
        reminder_date: str,
        title: str,
        description: str = None,
        priority: Literal["low", "medium", "high", "urgent"] = None,
        advance_notice_days: int = None
    ):
        """
        Create a wellness reminder for pet care activities.

        This method creates a new wellness reminder entry in the database for a specific pet.
        It validates the input parameters, generates a unique reminder ID, and calculates
        the notification date based on advance notice settings.
        """
        import secrets
        import hashlib
        from datetime import datetime, timedelta

        # Validate required parameters are not empty
        if not pet_id or not pet_id.strip():
            raise ValueError("pet_id cannot be empty")
        if not reminder_type or not reminder_type.strip():
            raise ValueError("reminder_type cannot be empty")
        if not reminder_date or not reminder_date.strip():
            raise ValueError("reminder_date cannot be empty")
        if not title or not title.strip():
            raise ValueError("title cannot be empty")

        # Validate reminder_type against enum values (safety protection)
        valid_reminder_types = ["vaccination", "medication", "checkup", "grooming", 
                               "flea_treatment", "deworming", "dental_care", "exercise", "diet_review"]
        if reminder_type not in valid_reminder_types:
            raise ValueError(f"Invalid reminder_type. Must be one of: {', '.join(valid_reminder_types)}")

        # Validate priority against enum values if provided (safety protection)
        if priority is not None:
            valid_priorities = ["low", "medium", "high", "urgent"]
            if priority not in valid_priorities:
                raise ValueError(f"Invalid priority. Must be one of: {', '.join(valid_priorities)}")

        # Parse and validate reminder_date format (yyyy-mm-dd)
        try:
            parsed_reminder_date = datetime.strptime(reminder_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError("reminder_date must be in yyyy-mm-dd format")

        # Validate advance_notice_days if provided
        if advance_notice_days is not None:
            if not isinstance(advance_notice_days, int) or advance_notice_days < 0:
                raise ValueError("advance_notice_days must be a non-negative integer")

        # Generate unique reminder_id
        prefix = "reminder_"
        reminder_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for creation_timestamp
        creation_timestamp = datetime.now()

        # Calculate notification_date based on advance_notice_days
        # If advance_notice_days is provided, notification should be sent that many days before reminder_date
        # Otherwise, notification_date is same as reminder_date
        if advance_notice_days is not None and advance_notice_days > 0:
            notification_date = parsed_reminder_date - timedelta(days=advance_notice_days)
        else:
            notification_date = parsed_reminder_date

        # Access the database
        db = self.db

        # Get the wellness_reminder table, initialize if it doesn't exist
        wellness_reminder_table = getattr(db, "wellness_reminder", None)
        if wellness_reminder_table is None:
            wellness_reminder_table = {}

        # Create new WellnessReminder instance
        new_reminder = WellnessReminder(
            reminder_id=reminder_id,
            pet_id=pet_id,
            reminder_type=reminder_type,
            reminder_date=parsed_reminder_date,
            title=title,
            description=description,
            priority=priority,
            advance_notice_days=advance_notice_days,
            status="active",  # Default status for new reminders
            creation_timestamp=creation_timestamp
        )

        # Add the new reminder to the table
        wellness_reminder_table[reminder_id] = new_reminder

        # Save the updated table back to the database
        setattr(db, "wellness_reminder", wellness_reminder_table)

        # Return the result with required fields
        return {
            "reminder_id": reminder_id,
            "creation_timestamp": creation_timestamp.strftime("%Y-%m-%d %H:%M:%S"),
            "notification_date": notification_date.strftime("%Y-%m-%d")
        }

    @is_tool()
    def get_vaccination_history(self, pet_id: str):
        # Get the database instance
        db = self.db

        # Retrieve the vaccination_record table from the database
        vaccination_table = getattr(db, "vaccination_record", None)

        # Check if the vaccination_record table exists
        if vaccination_table is None:
            raise KeyError(f"vaccination_record table does not exist in the database")

        # Initialize the list to store vaccination records for the specified pet
        pet_vaccinations = []

        # Iterate through all vaccination records in the table
        for vacc_id, vaccination in vaccination_table.items():
            # Check if the vaccination record belongs to the specified pet
            if vaccination.pet_id == pet_id:
                # Create a dictionary representation of the vaccination record
                # Convert date objects to string format "yyyy-mm-dd"
                vaccination_dict = {
                    "vaccination_id": vaccination.vaccination_id,
                    "vaccine_name": vaccination.vaccine_name,
                    "administration_date": vaccination.administration_date.strftime("%Y-%m-%d"),
                    "next_due_date": vaccination.next_due_date.strftime("%Y-%m-%d")
                }

                # Add optional fields if they exist
                if vaccination.vaccine_type is not None:
                    vaccination_dict["vaccine_type"] = vaccination.vaccine_type

                if vaccination.veterinarian_name is not None:
                    vaccination_dict["veterinarian_name"] = vaccination.veterinarian_name

                if vaccination.batch_number is not None:
                    vaccination_dict["batch_number"] = vaccination.batch_number

                if vaccination.clinic_id is not None:
                    vaccination_dict["clinic_id"] = vaccination.clinic_id

                # Add creation timestamp in "yyyy-mm-dd HH:MM:SS" format
                vaccination_dict["creation_timestamp"] = vaccination.creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

                # Append the vaccination record to the list
                pet_vaccinations.append(vaccination_dict)

        # Sort vaccinations by administration_date in descending order (most recent first)
        # This provides a more useful chronological view of the pet's vaccination history
        pet_vaccinations.sort(key=lambda x: x["administration_date"], reverse=True)

        # Calculate the total count of vaccination records
        total_count = len(pet_vaccinations)

        # Return the complete vaccination history with total count
        return {
            "vaccinations": pet_vaccinations,
            "total_count": total_count
        }
