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

class HealthcareTelemedicineTools(ToolKitBase):
    """All tools for healthcare_telemedicine."""
    
    db: HealthcareTelemedicineDB
    
    def __init__(self, db: HealthcareTelemedicineDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def delete_medical_document(self, document_id: str, patient_id: str, deletion_reason: str = None):
        # Import required modules
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Get the medical_document table from the database
        medical_document_table = getattr(db, "medical_document", None)

        # Validate that the medical_document table exists
        if medical_document_table is None:
            raise ValueError("Medical document table does not exist in the database")

        # Check if the document exists in the database
        if document_id not in medical_document_table:
            raise ValueError(f"Document with ID '{document_id}' does not exist")

        # Retrieve the document from the table
        document = medical_document_table[document_id]

        # Verify that the document belongs to the specified patient
        # This ensures patients can only delete their own documents
        if document.patient_id != patient_id:
            raise ValueError(f"Document '{document_id}' does not belong to patient '{patient_id}'")

        # Check if the document has already been deleted
        if document.deleted_at is not None:
            raise ValueError(f"Document '{document_id}' has already been deleted")

        # Record the deletion timestamp in yyyy-mm-dd HH:MM:SS format
        deleted_at = datetime.now()
        deleted_at_str = deleted_at.strftime("%Y-%m-%d %H:%M:%S")

        # Update the document with deletion information
        document.deleted_at = deleted_at
        if deletion_reason:
            document.deletion_reason = deletion_reason

        # Persist the updated document back to the database
        # The document is marked as deleted but not physically removed
        # This allows for audit trails and potential recovery
        medical_document_table[document_id] = document
        setattr(db, "medical_document", medical_document_table)

        # Return the deletion status and timestamp
        return {
            "deletion_status": True,
            "deleted_at": deleted_at_str
        }

    @is_tool()
    def submit_provider_rating(
        self,
        patient_id: str,
        provider_id: str,
        appointment_id: str,
        overall_rating: int,
        communication_rating: int = None,
        professionalism_rating: int = None,
        written_review: str = None,
        would_recommend: bool = None
    ):
        """
        Submit a rating and review for a healthcare provider after an appointment.

        This method validates the input parameters, checks that the appointment is completed
        and that the patient has not already submitted a rating, then creates and stores
        a new provider rating record.
        """
        from datetime import datetime
        import secrets
        import hashlib

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

        # Validate overall_rating is an integer between 1 and 5
        if not isinstance(overall_rating, int):
            raise ValueError("overall_rating must be an integer")
        if overall_rating < 1 or overall_rating > 5:
            raise ValueError("overall_rating must be between 1 and 5 stars")

        # Validate optional rating parameters if provided
        if communication_rating is not None:
            if not isinstance(communication_rating, int):
                raise ValueError("communication_rating must be an integer")
            if communication_rating < 1 or communication_rating > 5:
                raise ValueError("communication_rating must be between 1 and 5 stars")

        if professionalism_rating is not None:
            if not isinstance(professionalism_rating, int):
                raise ValueError("professionalism_rating must be an integer")
            if professionalism_rating < 1 or professionalism_rating > 5:
                raise ValueError("professionalism_rating must be between 1 and 5 stars")

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

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

        # Access the database
        db = self.db

        # Get the provider_rating table
        provider_rating_table = getattr(db, "provider_rating", None)
        if provider_rating_table is None:
            # Initialize empty table if it doesn't exist
            provider_rating_table = {}
            setattr(db, "provider_rating", provider_rating_table)

        # Check if patient has already submitted a rating for this appointment
        # Iterate through existing ratings to find duplicates
        for rating_id, rating in provider_rating_table.items():
            if (rating.patient_id == patient_id and 
                rating.appointment_id == appointment_id):
                raise ValueError(
                    f"Patient {patient_id} has already submitted a rating for appointment {appointment_id}"
                )

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

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

        # Get current timestamp
        submitted_at = datetime.now()

        # Create new ProviderRating instance
        new_rating = ProviderRating(
            rating_id=rating_id,
            patient_id=patient_id,
            provider_id=provider_id,
            appointment_id=appointment_id,
            overall_rating=overall_rating,
            communication_rating=communication_rating,
            professionalism_rating=professionalism_rating,
            written_review=written_review,
            would_recommend=would_recommend,
            submitted_at=submitted_at
        )

        # Add the new rating to the database
        provider_rating_table[rating_id] = new_rating
        setattr(db, "provider_rating", provider_rating_table)

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

    @is_tool()
    def list_patient_documents(
        self,
        patient_id: str,
        document_type: Literal["lab_result", "prescription", "imaging_report", "insurance_card", "medical_history", "vaccination_record", "all"] = "all",
        start_date: Optional[str] = None,
        end_date: Optional[str] = None
    ) -> dict:
        """
        Retrieve list of all medical documents for a patient with optional filtering by document type and date range.

        Args:
            patient_id: Unique identifier of the patient
            document_type: Filter documents by type, defaults to "all" to return all types
            start_date: Start date for filtering in yyyy-mm-dd format (optional)
            end_date: End date for filtering in yyyy-mm-dd format (optional)

        Returns:
            Dictionary containing:
            - documents: List of medical documents matching the criteria
            - total_count: Total number of documents found

        Raises:
            ValueError: If patient_id is empty, date format is invalid, or date range is invalid
        """
        from datetime import datetime

        # Validate patient_id is not empty
        if not patient_id or not patient_id.strip():
            raise ValueError("patient_id cannot be empty")

        # Parse and validate date parameters if provided
        start_datetime = None
        end_datetime = None

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

        if end_date:
            try:
                # Parse date string to datetime object (set time to end of day)
                end_datetime = datetime.strptime(end_date.strip(), "%Y-%m-%d")
                # Set to end of day (23:59:59) to include documents uploaded 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 if both dates are provided
        if start_datetime and end_datetime and start_datetime > end_datetime:
            raise ValueError("start_date cannot be after end_date")

        # Access database
        db = self.db
        medical_document_table = getattr(db, "medical_document", None)

        # If table doesn't exist or is empty, return empty result
        if not medical_document_table:
            return {
                "documents": [],
                "total_count": 0
            }

        # Filter documents based on criteria
        filtered_documents = []

        for doc_id, document in medical_document_table.items():
            # Skip if document doesn't belong to the specified patient
            if document.patient_id != patient_id:
                continue

            # Skip deleted documents (documents with deleted_at timestamp)
            if document.deleted_at is not None:
                continue

            # Filter by document_type if not "all"
            if document_type != "all" and document.document_type != document_type:
                continue

            # Filter by date range if specified
            if start_datetime and document.uploaded_at < start_datetime:
                continue

            if end_datetime and document.uploaded_at > end_datetime:
                continue

            # Document passes all filters, add to result list
            filtered_documents.append({
                "document_id": document.document_id,
                "document_type": document.document_type,
                "file_name": document.file_name,
                "file_size_bytes": document.file_size_bytes,
                "file_format": document.file_format,
                "description": document.description,
                "file_path": document.file_path,
                "upload_status": document.upload_status,
                "uploaded_at": document.uploaded_at.strftime("%Y-%m-%d %H:%M:%S")
            })

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

        # Return result with documents list and total count
        return {
            "documents": filtered_documents,
            "total_count": len(filtered_documents)
        }

    @is_tool()
    def update_patient_profile(
        self,
        patient_id: str,
        address: Optional[str] = None,
        emergency_contact_name: Optional[str] = None,
        emergency_contact_phone: Optional[str] = None,
        preferred_language: Optional[Literal["English", "Spanish", "French", "Chinese", "German"]] = None
    ) -> dict:
        """
        Update patient profile information including contact details and emergency contacts.

        This method updates the specified fields of a patient's profile in the database.
        Only the provided fields will be updated; fields not provided will remain unchanged.
        The patient must exist and have an active account status.

        Args:
            patient_id: Unique identifier of the patient
            address: Patient's residential address (optional)
            emergency_contact_name: Name of emergency contact person (optional)
            emergency_contact_phone: Phone number of emergency contact (optional)
            preferred_language: Patient's preferred language for communication (optional)
                Must be one of: English, Spanish, French, Chinese, German

        Returns:
            dict: Contains update_status (bool) and updated_at (str in yyyy-mm-dd HH:MM:SS format)

        Raises:
            ValueError: If patient_id is empty, patient not found, account is not active,
                       or preferred_language is not in the allowed enum values
        """
        from datetime import datetime

        # Validate patient_id is not empty
        if not patient_id or not patient_id.strip():
            raise ValueError("patient_id cannot be empty")

        # Get database instance
        db = self.db

        # Retrieve patient table from database
        patient_table = getattr(db, "patient", None)
        if patient_table is None:
            raise ValueError("Patient table not found in database")

        # Check if patient exists
        if patient_id not in patient_table:
            raise ValueError(f"Patient with patient_id '{patient_id}' not found")

        # Get the patient record
        patient = patient_table[patient_id]

        # Verify account is active (pre-condition check)
        if patient.account_status != "active":
            raise ValueError(f"Patient account is not active. Current status: {patient.account_status}")

        # Validate preferred_language if provided (enum constraint with safety protection)
        allowed_languages = ["English", "Spanish", "French", "Chinese", "German"]
        if preferred_language is not None:
            if preferred_language not in allowed_languages:
                raise ValueError(
                    f"Invalid preferred_language '{preferred_language}'. "
                    f"Must be one of: {', '.join(allowed_languages)}"
                )

        # Track if any field was actually updated
        fields_updated = False

        # Update address if provided
        if address is not None:
            patient.address = address
            fields_updated = True

        # Update emergency contact name if provided
        if emergency_contact_name is not None:
            patient.emergency_contact_name = emergency_contact_name
            fields_updated = True

        # Update emergency contact phone if provided
        if emergency_contact_phone is not None:
            patient.emergency_contact_phone = emergency_contact_phone
            fields_updated = True

        # Update preferred language if provided
        if preferred_language is not None:
            patient.preferred_language = preferred_language
            fields_updated = True

        # Update the updated_at timestamp
        current_time = datetime.now()
        patient.updated_at = current_time

        # Save the updated patient back to database
        patient_table[patient_id] = patient
        setattr(db, "patient", patient_table)

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

        # Return update status and timestamp
        return {
            "update_status": True,
            "updated_at": updated_at_str
        }

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

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

        # Parse and validate date parameters if provided
        start_datetime = None
        end_datetime = None

        if start_date:
            try:
                # Parse start_date string to datetime object (set time to 00:00:00)
                start_datetime = datetime.strptime(start_date.strip(), "%Y-%m-%d")
            except ValueError:
                raise ValueError(f"start_date must be in yyyy-mm-dd format, got: {start_date}")

        if end_date:
            try:
                # Parse end_date string to datetime object (set time to 23:59:59 for inclusive end)
                end_datetime = datetime.strptime(end_date.strip(), "%Y-%m-%d").replace(hour=23, minute=59, second=59)
            except ValueError:
                raise ValueError(f"end_date must be in yyyy-mm-dd format, got: {end_date}")

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

        # Get payment table from database
        payment_table = getattr(db, "payment", None)

        # Initialize result variables
        payments = []
        total_paid = 0.0
        transaction_count = 0

        # If payment table exists and has data, filter and process payments
        if payment_table:
            for payment_id, payment in payment_table.items():
                # Filter by patient_id (exact match for identifier)
                if payment.patient_id != patient_id:
                    continue

                # Filter by date range if provided
                payment_datetime = payment.processed_at

                # Apply start_date filter
                if start_datetime and payment_datetime < start_datetime:
                    continue

                # Apply end_date filter
                if end_datetime and payment_datetime > end_datetime:
                    continue

                # Payment matches all filters, add to results
                # Format processed_at to yyyy-mm-dd format for date field in response
                payment_date_str = payment_datetime.strftime("%Y-%m-%d")

                # Build payment transaction object
                payment_transaction = {
                    "payment_id": payment.payment_id,
                    "amount": payment.payment_amount,
                    "date": payment_date_str,
                    "status": payment.transaction_status,
                    "payment_method": payment.payment_method,
                    "appointment_id": payment.appointment_id
                }

                # Add optional fields if they exist
                if payment.card_last_four:
                    payment_transaction["card_last_four"] = payment.card_last_four
                if payment.receipt_number:
                    payment_transaction["receipt_number"] = payment.receipt_number

                payments.append(payment_transaction)
                total_paid += payment.payment_amount
                transaction_count += 1

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

        # Return payment history summary
        return {
            "payments": payments,
            "total_paid": round(total_paid, 2),  # Round to 2 decimal places for currency
            "transaction_count": transaction_count
        }

    @is_tool()
    def get_prescription_history(
        self,
        patient_id: str,
        start_date: str = None,
        end_date: str = None,
        status_filter: Literal["active", "completed", "cancelled", "all"] = "all"
    ):
        """
        Retrieve complete prescription history for a patient with optional date range filtering.

        Args:
            patient_id: Unique identifier of the patient
            start_date: Start date for filtering in yyyy-mm-dd format (optional)
            end_date: End date for filtering in yyyy-mm-dd format (optional)
            status_filter: Filter prescriptions by status (active/completed/cancelled/all)

        Returns:
            Dictionary containing:
            - prescriptions: List of prescription dictionaries
            - total_count: Total number of prescriptions matching the criteria

        Raises:
            ValueError: If patient_id is empty, date format is invalid, or date range is invalid
        """
        from datetime import datetime

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

        # Access the database
        db = self.db
        prescription_table = getattr(db, "prescription", None)

        # If prescription table doesn't exist or is empty, return empty results
        if prescription_table is None or len(prescription_table) == 0:
            return {
                "prescriptions": [],
                "total_count": 0
            }

        # Parse and validate date range if provided
        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 yyyy-mm-dd format")

        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 yyyy-mm-dd format")

        # 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")

        # Filter prescriptions based on criteria
        filtered_prescriptions = []

        for prescription_id, prescription in prescription_table.items():
            # Filter by patient_id (exact match for ID field)
            if prescription.patient_id != patient_id:
                continue

            # Filter by date range if specified
            if start_date_obj and prescription.prescribed_date < start_date_obj:
                continue

            if end_date_obj and prescription.prescribed_date > end_date_obj:
                continue

            # Filter by status if not "all"
            if status_filter != "all" and prescription.status != status_filter:
                continue

            # Convert prescription to dictionary format for return
            prescription_dict = {
                "prescription_id": prescription.prescription_id,
                "patient_id": prescription.patient_id,
                "provider_id": prescription.provider_id,
                "medication_name": prescription.medication_name,
                "prescribed_date": prescription.prescribed_date.strftime("%Y-%m-%d"),
                "status": prescription.status,
                "refills_remaining": prescription.refills_remaining,
                "created_at": prescription.created_at.strftime("%Y-%m-%d %H:%M:%S")
            }

            filtered_prescriptions.append(prescription_dict)

        # Sort prescriptions by prescribed_date in descending order (most recent first)
        filtered_prescriptions.sort(
            key=lambda x: datetime.strptime(x["prescribed_date"], "%Y-%m-%d"),
            reverse=True
        )

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

    @is_tool()
    def submit_payment(
        self,
        patient_id: str,
        appointment_id: str,
        payment_amount: float,
        payment_method: Literal["credit_card", "debit_card", "insurance", "health_savings_account"],
        card_last_four: str = None
    ) -> dict:
        """
        Submit payment for telehealth services or copay.

        This method processes a payment transaction for a patient's appointment,
        validates the payment information, generates a unique payment ID and receipt number,
        and stores the payment record in the database.

        Args:
            patient_id: Unique identifier of the patient making the payment
            appointment_id: Unique identifier of the appointment being paid for
            payment_amount: Amount to be paid (must be positive)
            payment_method: Method of payment (must be one of the enum values)
            card_last_four: Last four digits of payment card (optional, but recommended for card payments)

        Returns:
            dict: Payment transaction details including payment_id, transaction_status,
                  receipt_number, and processed_at timestamp

        Raises:
            ValueError: If payment_amount is not positive, or if required parameters are invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

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

        # Validate payment_method is one of the allowed enum values
        allowed_methods = ["credit_card", "debit_card", "insurance", "health_savings_account"]
        if payment_method not in allowed_methods:
            raise ValueError(
                f"payment_method must be one of {allowed_methods}, got '{payment_method}'"
            )

        # Validate card_last_four format if provided
        if card_last_four is not None:
            if not isinstance(card_last_four, str):
                raise ValueError("card_last_four must be a string")
            # Check if it's exactly 4 digits
            if not (len(card_last_four) == 4 and card_last_four.isdigit()):
                raise ValueError("card_last_four must be exactly 4 digits")

        # Access the database
        db = self.db

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

        # Generate unique payment_id using hash
        payment_prefix = "PAY-"
        payment_id = payment_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Ensure payment_id is unique (in case of collision)
        payment_table = getattr(db, "payment")
        while payment_id in payment_table:
            payment_id = payment_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Generate unique receipt_number
        receipt_prefix = "RCPT-"
        receipt_number = receipt_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Set transaction status as completed
        transaction_status = "completed"

        # Create Payment object
        new_payment = Payment(
            payment_id=payment_id,
            patient_id=patient_id,
            appointment_id=appointment_id,
            payment_amount=payment_amount,
            payment_method=payment_method,
            card_last_four=card_last_four,
            transaction_status=transaction_status,
            receipt_number=receipt_number,
            processed_at=processed_at
        )

        # Store payment in database
        payment_table[payment_id] = new_payment
        setattr(db, "payment", payment_table)

        # Format processed_at to required string format "yyyy-mm-dd HH:MM:SS"
        processed_at_str = processed_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return payment transaction details
        return {
            "payment_id": payment_id,
            "transaction_status": transaction_status,
            "receipt_number": receipt_number,
            "processed_at": processed_at_str
        }

    @is_tool()
    def add_pharmacy_to_profile(self, patient_id: str, pharmacy_name: str, pharmacy_address: str, pharmacy_phone: str, is_primary: bool = False):
        import secrets
        import hashlib
        from datetime import datetime

        # Input validation - check for empty or whitespace-only strings
        if not patient_id or not patient_id.strip():
            raise ValueError("patient_id cannot be empty")
        if not pharmacy_name or not pharmacy_name.strip():
            raise ValueError("pharmacy_name cannot be empty")
        if not pharmacy_address or not pharmacy_address.strip():
            raise ValueError("pharmacy_address cannot be empty")
        if not pharmacy_phone or not pharmacy_phone.strip():
            raise ValueError("pharmacy_phone cannot be empty")

        # Access the database through self.db (database instance is already available)
        db = self.db

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

        # Generate unique pharmacy_id using secure hash
        prefix = "PHARM-"
        pharmacy_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Get current timestamp
        added_at = datetime.now()

        # Create new Pharmacy instance (Pharmacy class is already imported at file header)
        new_pharmacy = Pharmacy(
            pharmacy_id=pharmacy_id,
            patient_id=patient_id.strip(),
            pharmacy_name=pharmacy_name.strip(),
            pharmacy_address=pharmacy_address.strip(),
            pharmacy_phone=pharmacy_phone.strip(),
            is_primary=is_primary,
            added_at=added_at
        )

        # Add the new pharmacy to the table
        pharmacy_table[pharmacy_id] = new_pharmacy

        # Save the updated table back to the database
        setattr(db, 'pharmacy', pharmacy_table)

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

        # Return the result
        return {
            "pharmacy_id": pharmacy_id,
            "added_at": added_at_str
        }

    @is_tool()
    def get_provider_availability(
        self,
        provider_id: str,
        start_date: str,
        end_date: str,
        appointment_type: Literal["video_consultation", "phone_consultation", "in_person"] = None
    ):
        """
        Retrieve available appointment time slots for a specific healthcare provider within a date range.

        This method:
        1. Validates the provider exists in the database
        2. Validates date format and range
        3. Retrieves all appointments for the provider in the specified date range
        4. Calculates available time slots based on existing appointments
        5. Filters by appointment type if specified
        6. Returns list of available slots with total count

        Args:
            provider_id: Unique identifier of the healthcare provider
            start_date: Start date for availability search in yyyy-mm-dd format
            end_date: End date for availability search in yyyy-mm-dd format
            appointment_type: Optional filter for specific appointment type

        Returns:
            Dictionary containing:
            - available_slots: List of available time slot dictionaries
            - total_slots: Total number of available slots

        Raises:
            ValueError: If provider doesn't exist, dates are invalid, or date range is invalid
        """
        from datetime import datetime, timedelta
        import secrets
        import hashlib

        # Access database
        db = self.db

        # Validate provider exists
        healthcare_provider_table = getattr(db, "healthcare_provider", None)
        if healthcare_provider_table is None:
            raise ValueError("Healthcare provider table not found in database")

        provider = healthcare_provider_table.get(provider_id)
        if provider is None:
            raise ValueError(f"Provider with ID '{provider_id}' does not exist")

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

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

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

        # Set time boundaries for the search range
        # Start from beginning of start_date, end at end of end_date
        search_start = start_dt.replace(hour=0, minute=0, second=0, microsecond=0)
        search_end = end_dt.replace(hour=23, minute=59, second=59, microsecond=999999)

        # Get appointment table
        appointment_table = getattr(db, "appointment", None)
        if appointment_table is None:
            # No appointments table means all slots are available
            appointment_table = {}

        # Retrieve all appointments for this provider in the date range
        # Filter by provider_id, date range, and non-cancelled status
        provider_appointments = []
        for apt_id, apt in appointment_table.items():
            if apt.provider_id == provider_id:
                # Check if appointment is in date range
                if search_start <= apt.appointment_datetime <= search_end:
                    # Only consider scheduled appointments (not cancelled or no_show)
                    if apt.status in ["scheduled", "completed", "rescheduled"]:
                        provider_appointments.append(apt)

        # Sort appointments by datetime
        provider_appointments.sort(key=lambda x: x.appointment_datetime)

        # Define working hours (9 AM to 5 PM) and slot duration
        work_start_hour = 9
        work_end_hour = 17
        slot_duration_minutes = provider.average_consultation_minutes or 30

        # Generate all possible time slots for each day in the range
        available_slots = []
        current_date = start_dt.date()
        end_date_obj = end_dt.date()

        while current_date <= end_date_obj:
            # Generate slots for this day
            day_start = datetime.combine(current_date, datetime.min.time()).replace(
                hour=work_start_hour, minute=0, second=0
            )
            day_end = datetime.combine(current_date, datetime.min.time()).replace(
                hour=work_end_hour, minute=0, second=0
            )

            # Generate time slots for the day
            current_slot_start = day_start
            while current_slot_start < day_end:
                slot_end = current_slot_start + timedelta(minutes=slot_duration_minutes)

                # Check if slot_end exceeds working hours
                if slot_end > day_end:
                    break

                # Check if this slot conflicts with any existing appointment
                is_available = True
                for apt in provider_appointments:
                    apt_start = apt.appointment_datetime
                    apt_end = apt_start + timedelta(minutes=apt.duration_minutes or 30)

                    # Check for overlap: slot overlaps with appointment if:
                    # slot_start < apt_end AND slot_end > apt_start
                    if current_slot_start < apt_end and slot_end > apt_start:
                        is_available = False
                        break

                # If appointment_type filter is specified, we mark slot as available
                # (type filtering is logical - all slots are available for any type unless booked)
                if is_available:
                    # Generate unique slot_id
                    slot_id_hash = hashlib.sha256(
                        f"{provider_id}_{current_slot_start.isoformat()}".encode()
                    ).hexdigest()[:10]
                    slot_id = f"SLOT-{slot_id_hash}"

                    slot = {
                        "slot_id": slot_id,
                        "start_time": current_slot_start.strftime("%Y-%m-%d %H:%M:%S"),
                        "end_time": slot_end.strftime("%Y-%m-%d %H:%M:%S")
                    }
                    available_slots.append(slot)

                # Move to next slot
                current_slot_start = slot_end

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

        # Note: appointment_type parameter is for filtering request type preference
        # Since slots themselves don't have inherent types (any slot can be used for any type),
        # we return all available slots. The appointment_type is used when booking.
        # If needed for business logic, it could filter slots but typically all slots
        # are available for all types unless provider has specific restrictions.

        # Return results
        return {
            "available_slots": available_slots,
            "total_slots": len(available_slots)
        }

    @is_tool()
    def create_patient_account(
        self,
        first_name: str,
        last_name: str,
        email: str,
        phone_number: str,
        date_of_birth: str,
        password: str
    ):
        """
        Create a new patient account with basic registration information.

        This method validates the input parameters, checks for duplicate email addresses,
        generates a unique patient ID, hashes the password, and creates a new patient record
        in the database with pending verification status.
        """
        # Import required modules inside the method
        from datetime import datetime, date
        import hashlib
        import secrets
        import re

        # Validate required parameters are not empty
        if not first_name or not first_name.strip():
            raise ValueError("First name cannot be empty")
        if not last_name or not last_name.strip():
            raise ValueError("Last name cannot be empty")
        if not email or not email.strip():
            raise ValueError("Email cannot be empty")
        if not phone_number or not phone_number.strip():
            raise ValueError("Phone number cannot be empty")
        if not date_of_birth or not date_of_birth.strip():
            raise ValueError("Date of birth cannot be empty")
        if not password or not password.strip():
            raise ValueError("Password cannot be empty")

        # Validate email format
        email_pattern = r'^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$'
        if not re.match(email_pattern, email.strip()):
            raise ValueError("Invalid email format")

        # Validate date of birth format (yyyy-mm-dd)
        try:
            dob_date = datetime.strptime(date_of_birth.strip(), "%Y-%m-%d").date()
            # Check if date of birth is not in the future
            if dob_date > date.today():
                raise ValueError("Date of birth cannot be in the future")
        except ValueError as e:
            if "does not match format" in str(e) or "time data" in str(e):
                raise ValueError("Date of birth must be in yyyy-mm-dd format")
            raise

        # Validate password strength (basic requirements)
        if len(password) < 8:
            raise ValueError("Password must be at least 8 characters long")

        # Get database instance
        db = self.db

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

        # Check if email already exists (case-insensitive comparison)
        normalized_email = email.strip().lower()
        for existing_patient in patient_table.values():
            if existing_patient.email.lower() == normalized_email:
                raise ValueError("Email address is already registered in the system")

        # Generate unique patient ID with PAT prefix
        prefix = "PAT-"
        patient_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Ensure the generated ID is unique (extremely unlikely collision, but safe to check)
        while patient_id in patient_table:
            patient_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Hash the password using SHA256
        password_hash = hashlib.sha256(password.encode('utf-8')).hexdigest()

        # Get current timestamp
        created_at = datetime.now()

        # Create new patient record
        new_patient = Patient(
            patient_id=patient_id,
            first_name=first_name.strip(),
            last_name=last_name.strip(),
            email=email.strip(),
            phone_number=phone_number.strip(),
            date_of_birth=dob_date,
            password_hash=password_hash,
            account_status="pending_verification",
            email_verified=False,
            two_factor_enabled=False,
            created_at=created_at
        )

        # Add patient to database
        patient_table[patient_id] = new_patient
        setattr(db, "patient", patient_table)

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

        # Return patient account information
        return {
            "patient_id": patient_id,
            "account_status": "pending_verification",
            "created_at": created_at_str
        }

    @is_tool()
    def cancel_appointment(self, appointment_id: str, cancellation_reason: str, cancelled_by: Literal["patient", "provider", "system"]):
        """
        Cancel an existing scheduled appointment and release the time slot

        Args:
            appointment_id: Unique identifier of the appointment to cancel
            cancellation_reason: Reason for cancelling the appointment
            cancelled_by: Who initiated the cancellation (must be one of: patient, provider, system)

        Returns:
            dict: Contains cancellation_status, refund_eligible, and cancelled_at timestamp

        Raises:
            ValueError: If appointment doesn't exist, not in scheduled status, or invalid parameters
        """
        from datetime import datetime

        # Access the database
        db = self.db

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

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

        # Validate cancelled_by enum value (additional safety check)
        valid_cancelled_by = ["patient", "provider", "system"]
        if cancelled_by not in valid_cancelled_by:
            raise ValueError(f"cancelled_by must be one of {valid_cancelled_by}, got: {cancelled_by}")

        # Retrieve appointment table from database
        appointment_table = getattr(db, "appointment", None)
        if appointment_table is None:
            raise ValueError("Appointment table does not exist in database")

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

        # Get the appointment object
        appointment = appointment_table[appointment_id]

        # Verify appointment is in scheduled status (pre-condition check)
        if appointment.status != "scheduled":
            raise ValueError(f"Appointment must be in 'scheduled' status to cancel. Current status: '{appointment.status}'")

        # Calculate cancellation timestamp
        cancelled_at = datetime.now()

        # Determine refund eligibility based on cancellation timing
        # Refund eligible if cancelled more than 24 hours before appointment
        time_until_appointment = appointment.appointment_datetime - cancelled_at
        hours_until_appointment = time_until_appointment.total_seconds() / 3600
        refund_eligible = hours_until_appointment > 24

        # Update appointment status and cancellation details
        appointment.status = "cancelled"
        appointment.cancellation_reason = cancellation_reason
        appointment.cancelled_by = cancelled_by
        appointment.cancelled_at = cancelled_at

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

        # Format cancelled_at timestamp as string in required format
        cancelled_at_str = cancelled_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return cancellation result
        return {
            "cancellation_status": True,
            "refund_eligible": refund_eligible,
            "cancelled_at": cancelled_at_str
        }

    @is_tool()
    def get_outstanding_balance(self, patient_id: str):
        """
        Retrieve current outstanding balance for a patient.

        Args:
            patient_id: Unique identifier of the patient

        Returns:
            dict: Contains total_balance, itemized_charges, and oldest_charge_date

        Raises:
            ValueError: If patient_id is invalid or payment table not found
        """
        # Validate input parameter
        if not patient_id or not isinstance(patient_id, str):
            raise ValueError("patient_id must be a non-empty string")

        # Access the database
        db = self.db

        # Get payment table from database
        payment_table = getattr(db, "payment", None)
        if payment_table is None:
            raise ValueError("Payment table not found in database")

        # Initialize return values
        total_balance = 0.0
        itemized_charges = []
        oldest_charge_date = None

        # Iterate through all payment records to find unpaid charges for the patient
        for payment_id, payment in payment_table.items():
            # Check if the payment belongs to the specified patient
            if payment.patient_id != patient_id:
                continue

            # Check if the payment is not completed (outstanding)
            # Assuming non-completed payments are outstanding charges
            if payment.transaction_status != "completed":
                # Add to total balance
                total_balance += payment.payment_amount

                # Format the processed_at datetime to yyyy-mm-dd format for date comparison
                charge_date_str = payment.processed_at.strftime("%Y-%m-%d")

                # Create itemized charge entry
                charge_item = {
                    "charge_id": payment.payment_id,
                    "description": f"Payment for appointment {payment.appointment_id}",
                    "amount": payment.payment_amount,
                    "date": charge_date_str
                }
                itemized_charges.append(charge_item)

                # Track the oldest charge date
                if oldest_charge_date is None:
                    oldest_charge_date = payment.processed_at
                else:
                    if payment.processed_at < oldest_charge_date:
                        oldest_charge_date = payment.processed_at

        # Format oldest_charge_date to yyyy-mm-dd string format
        oldest_charge_date_str = oldest_charge_date.strftime("%Y-%m-%d") if oldest_charge_date else None

        # Sort itemized_charges by date (oldest first)
        itemized_charges.sort(key=lambda x: x["date"])

        # Return the outstanding balance details
        return {
            "total_balance": total_balance,
            "itemized_charges": itemized_charges,
            "oldest_charge_date": oldest_charge_date_str
        }

    @is_tool()
    def get_patient_profile(self, patient_id: str):
        """
        Retrieve complete patient profile information including personal details and account settings.

        This method fetches a patient's profile from the database using their unique patient_id.
        It returns all relevant profile information including personal details, contact information,
        and account settings.

        Args:
            patient_id: Unique identifier of the patient (e.g., "PAT-20240115-001")

        Returns:
            dict: A dictionary containing the patient's complete profile information with keys:
                - patient_id: Unique identifier of the patient
                - first_name: Patient's first name
                - last_name: Patient's last name
                - email: Patient's email address
                - phone_number: Patient's phone number
                - date_of_birth: Patient's date of birth in yyyy-mm-dd format
                - address: Patient's residential address
                - account_status: Current account status

        Raises:
            ValueError: If patient_id is empty/None or if patient is not found in database
        """
        # Validate input parameter
        if not patient_id or not isinstance(patient_id, str) or not patient_id.strip():
            raise ValueError("patient_id must be a non-empty string")

        # Access the database
        db = self.db

        # Retrieve the patient table from database
        patient_table = getattr(db, "patient", None)

        # Check if patient table exists in database
        if patient_table is None:
            raise ValueError("Patient table not found in database")

        # Retrieve the specific patient record using patient_id
        patient = patient_table.get(patient_id)

        # Check if patient exists
        if patient is None:
            raise ValueError(f"Patient with patient_id '{patient_id}' not found")

        # Convert date_of_birth from date object to string format yyyy-mm-dd
        date_of_birth_str = patient.date_of_birth.strftime("%Y-%m-%d")

        # Construct and return the patient profile dictionary
        # Only include the fields specified in the tool schema returns
        profile = {
            "patient_id": patient.patient_id,
            "first_name": patient.first_name,
            "last_name": patient.last_name,
            "email": patient.email,
            "phone_number": patient.phone_number,
            "date_of_birth": date_of_birth_str,
            "address": patient.address if patient.address else "",
            "account_status": patient.account_status
        }

        return profile

    @is_tool()
    def add_medication_to_list(
        self,
        patient_id: str,
        medication_name: str,
        dosage: str,
        frequency: str,
        start_date: str,
        prescribing_provider: str = None,
        instructions: str = None
    ):
        """
        Add a new medication to patient's current medication list with dosage and frequency information.

        This method creates a new medication entry in the database for the specified patient.
        It validates input parameters, generates a unique medication ID, and stores the medication
        with all relevant details including dosage, frequency, and optional instructions.

        Args:
            patient_id: Unique identifier of the patient
            medication_name: Name of the medication
            dosage: Dosage amount and unit (e.g., "10mg")
            frequency: How often the medication should be taken (e.g., "Once daily")
            start_date: Date when medication was started in yyyy-mm-dd format
            prescribing_provider: Optional name of the provider who prescribed the medication
            instructions: Optional special instructions for taking the medication

        Returns:
            Dictionary containing:
                - medication_id: Unique identifier for the medication entry
                - added_at: Timestamp when medication was added in yyyy-mm-dd HH:MM:SS format

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

        # Validate required parameters are not empty
        if not patient_id or not isinstance(patient_id, str) or not patient_id.strip():
            raise ValueError("patient_id is required and 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 is required and must be a non-empty string")

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

        if not frequency or not isinstance(frequency, str) or not frequency.strip():
            raise ValueError("frequency is required and 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 is required and must be a non-empty string")

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

        # Generate unique medication ID
        medication_id = "MED-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Access the database
        db = self.db

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

        # Get the medication table
        medication_table = getattr(db, "medication")

        # Create new medication entry
        new_medication = Medication(
            medication_id=medication_id,
            patient_id=patient_id.strip(),
            medication_name=medication_name.strip(),
            dosage=dosage.strip(),
            frequency=frequency.strip(),
            start_date=parsed_start_date,
            prescribing_provider=prescribing_provider.strip() if prescribing_provider else None,
            instructions=instructions.strip() if instructions else None,
            is_active=True,  # New medications are active by default
            discontinuation_date=None,
            discontinuation_reason=None,
            added_at=added_at,
            updated_at=None,
            removed_at=None
        )

        # Add medication to the table
        medication_table[medication_id] = new_medication

        # Update the database with the modified table
        setattr(db, "medication", medication_table)

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

    @is_tool()
    def send_secure_message(
        self,
        patient_id: str,
        provider_id: str,
        subject: str,
        message_body: str,
        priority: Literal["low", "normal", "high", "urgent"] = "normal",
        attachment_ids: Optional[List[str]] = None
    ) -> dict:
        """
        Send a secure message to healthcare provider through the patient portal.

        This method creates a new secure message from a patient to a provider and stores it
        in the database. It also handles optional attachments by creating message_attachment
        records.

        Args:
            patient_id: Unique identifier of the patient sending the message
            provider_id: Unique identifier of the provider receiving the message
            subject: Subject line of the message
            message_body: Content of the message
            priority: Priority level of the message (low, normal, high, urgent)
            attachment_ids: Optional list of document IDs to attach to the message

        Returns:
            dict containing:
                - message_id: Unique identifier for the sent message
                - sent_at: Timestamp when message was sent (yyyy-mm-dd HH:MM:SS format)
                - delivery_status: Status of message delivery

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

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

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

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

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

        # Validate priority parameter with enum constraint
        valid_priorities = ["low", "normal", "high", "urgent"]
        if priority not in valid_priorities:
            raise ValueError(f"priority must be one of {valid_priorities}, got '{priority}'")

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

        # Access the database
        db = self.db

        # Get secure_message and message_attachment tables
        secure_message_table = getattr(db, "secure_message", None)
        message_attachment_table = getattr(db, "message_attachment", None)

        if secure_message_table is None:
            secure_message_table = {}
            setattr(db, "secure_message", secure_message_table)

        if message_attachment_table is None:
            message_attachment_table = {}
            setattr(db, "message_attachment", message_attachment_table)

        # Generate unique message_id
        message_id_prefix = "MSG-"
        message_id = message_id_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Generate unique thread_id (for potential future thread management)
        thread_id_prefix = "THREAD-"
        thread_id = thread_id_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        sent_at = datetime.now()

        # Create new SecureMessage object
        new_message = SecureMessage(
            message_id=message_id,
            thread_id=thread_id,
            patient_id=patient_id,
            provider_id=provider_id,
            sender_type="patient",  # Patient is sending the message
            subject=subject,
            message_body=message_body,
            priority=priority,
            is_read=False,  # New message is unread
            delivery_status="delivered",  # Message is delivered upon creation
            sent_at=sent_at,
            read_at=None  # Not read yet
        )

        # Store the message in the database
        secure_message_table[message_id] = new_message
        setattr(db, "secure_message", secure_message_table)

        # Handle attachments if provided
        if attachment_ids:
            for document_id in attachment_ids:
                # Generate unique attachment_id
                attachment_id_prefix = "ATT-"
                attachment_id = attachment_id_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

                # Create MessageAttachment object
                new_attachment = MessageAttachment(
                    attachment_id=attachment_id,
                    message_id=message_id,
                    document_id=document_id
                )

                # Store the attachment in the database
                message_attachment_table[attachment_id] = new_attachment

            # Update the message_attachment table in the database
            setattr(db, "message_attachment", message_attachment_table)

        # Format sent_at timestamp to required format (yyyy-mm-dd HH:MM:SS)
        sent_at_str = sent_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "message_id": message_id,
            "sent_at": sent_at_str,
            "delivery_status": "delivered"
        }

    @is_tool()
    def enable_two_factor_authentication(self, patient_id: str, authentication_method: Literal["sms", "email", "authenticator_app"], phone_number: str = None):
        """
        Enable two-factor authentication for patient account security.

        This method enables 2FA for a patient account using the specified authentication method.
        It validates prerequisites, generates necessary setup data (QR codes for authenticator apps,
        backup codes), and updates the patient record.
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

        # Retrieve the patient table
        patient_table = getattr(db, "patient", None)
        if patient_table is None:
            raise ValueError("Patient table not found in database")

        # Find the patient by patient_id (exact match for ID)
        patient = patient_table.get(patient_id)
        if patient is None:
            raise ValueError(f"Patient with ID '{patient_id}' not found")

        # Check pre-condition: 2FA must not already be enabled
        if patient.two_factor_enabled:
            raise ValueError(f"Two-factor authentication is already enabled for patient '{patient_id}'")

        # Validate phone_number requirement for SMS method
        if authentication_method == "sms":
            if not phone_number:
                raise ValueError("Phone number is required when authentication method is 'sms'")
            # Update patient's phone number if provided
            patient.phone_number = phone_number

        # Generate backup codes (5 codes for recovery purposes)
        backup_codes = []
        for _ in range(5):
            # Generate a secure random backup code (8 characters alphanumeric)
            code = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:8].upper()
            backup_codes.append(code)

        # Generate QR code URL for authenticator app setup
        qr_code_url = None
        if authentication_method == "authenticator_app":
            # Generate QR code URL with patient_id embedded
            qr_code_url = f"https://telehealth.example.com/2fa/qr/{patient_id}"

        # Update patient record with 2FA settings
        current_time = datetime.now()
        patient.two_factor_enabled = True
        patient.two_factor_method = authentication_method
        patient.updated_at = current_time

        # Save the updated patient record back to the database
        patient_table[patient_id] = patient
        setattr(db, "patient", patient_table)

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

        # Prepare and return the result
        result = {
            "setup_status": True,
            "qr_code_url": qr_code_url,
            "backup_codes": backup_codes,
            "enabled_at": enabled_at_str
        }

        return result

    @is_tool()
    def schedule_appointment(
        self,
        patient_id: str,
        provider_id: str,
        appointment_datetime: str,
        appointment_type: Literal["video_consultation", "phone_consultation", "in_person"],
        reason_for_visit: str,
        duration_minutes: int = 30
    ):
        """
        Schedule a new telehealth appointment with a healthcare provider for a specific time slot.

        This method creates a new appointment record in the database after validating:
        1. All required parameters are provided and valid
        2. The appointment datetime is in the correct format and in the future
        3. The time slot is available (no conflicting appointments for the provider)
        4. The appointment type is one of the allowed values

        Args:
            patient_id: Unique identifier of the patient
            provider_id: Unique identifier of the healthcare provider
            appointment_datetime: Scheduled date and time in "yyyy-mm-dd HH:MM:SS" format
            appointment_type: Type of appointment (must be one of the enum values)
            reason_for_visit: Brief description of the reason for the appointment
            duration_minutes: Expected duration in minutes (default: 30)

        Returns:
            dict: Contains appointment_id, confirmation_number, status, and created_at

        Raises:
            ValueError: If parameters are invalid or time slot is not available
        """
        from datetime import datetime, timedelta
        import secrets
        import hashlib

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

        # Validate appointment_type is in allowed enum values
        allowed_types = ["video_consultation", "phone_consultation", "in_person"]
        if appointment_type not in allowed_types:
            raise ValueError(f"appointment_type must be one of {allowed_types}, got '{appointment_type}'")

        # Validate and parse appointment_datetime
        try:
            appt_dt = 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)}")

        # Check if appointment datetime is in the future
        current_time = datetime.now()
        if appt_dt <= current_time:
            raise ValueError(f"appointment_datetime must be in the future, got '{appointment_datetime}' which is not after current time")

        # Validate duration_minutes is positive
        if duration_minutes <= 0:
            raise ValueError(f"duration_minutes must be positive, got {duration_minutes}")

        # Access the database
        db = self.db
        appointments = getattr(db, "appointment", None)

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

        # Check for time slot availability
        # Calculate the end time of the new appointment
        appt_end_time = appt_dt + timedelta(minutes=duration_minutes)

        # Check for conflicts with existing appointments for the same provider
        for existing_appt in appointments.values():
            # Only check appointments that are scheduled (not cancelled or completed)
            if existing_appt.provider_id == provider_id and existing_appt.status == "scheduled":
                existing_start = existing_appt.appointment_datetime
                existing_end = existing_start + timedelta(minutes=existing_appt.duration_minutes)

                # Check for time overlap
                # Two appointments overlap if one starts before the other ends
                if (appt_dt < existing_end and appt_end_time > existing_start):
                    raise ValueError(
                        f"Time slot not available: conflicts with existing appointment {existing_appt.appointment_id} "
                        f"from {existing_start.strftime('%Y-%m-%d %H:%M:%S')} to {existing_end.strftime('%Y-%m-%d %H:%M:%S')}"
                    )

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

        # Generate confirmation number
        confirmation_number = "CONF-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:6].upper()

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

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

        # Create new appointment instance
        new_appointment = Appointment(
            appointment_id=appointment_id,
            patient_id=patient_id,
            provider_id=provider_id,
            appointment_datetime=appt_dt,
            appointment_type=appointment_type,
            reason_for_visit=reason_for_visit,
            duration_minutes=duration_minutes,
            status="scheduled",
            confirmation_number=confirmation_number,
            created_at=created_at
        )

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

        # Return appointment details
        return {
            "appointment_id": appointment_id,
            "confirmation_number": confirmation_number,
            "status": "scheduled",
            "created_at": created_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def update_notification_preferences(
        self,
        patient_id: str,
        appointment_reminders: bool = None,
        message_notifications: bool = None,
        medication_reminders: bool = None,
        lab_result_notifications: bool = None,
        notification_channels: List[Literal["email", "sms", "push_notification", "in_app"]] = None
    ):
        from datetime import datetime
        import secrets
        import hashlib

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

        # Validate notification_channels if provided
        valid_channels = {"email", "sms", "push_notification", "in_app"}
        if notification_channels is not None:
            if not isinstance(notification_channels, list):
                raise ValueError("notification_channels must be a list")
            for channel in notification_channels:
                if channel not in valid_channels:
                    raise ValueError(f"Invalid notification channel: {channel}. Must be one of {valid_channels}")

        # Access database
        db = self.db

        # Get existing notification_preference table
        notification_preference_table = getattr(db, "notification_preference", None)
        if notification_preference_table is None:
            notification_preference_table = {}

        # Get existing notification_channel table
        notification_channel_table = getattr(db, "notification_channel", None)
        if notification_channel_table is None:
            notification_channel_table = {}

        # Find existing preference record for this patient
        existing_preference = None
        preference_id = None
        for pref_id, pref in notification_preference_table.items():
            if pref.patient_id == patient_id:
                existing_preference = pref
                preference_id = pref_id
                break

        # Get current timestamp
        current_time = datetime.now()

        # If preference exists, update it; otherwise create new one
        if existing_preference:
            # Update only the fields that are provided (not None)
            if appointment_reminders is not None:
                existing_preference.appointment_reminders = appointment_reminders
            if message_notifications is not None:
                existing_preference.message_notifications = message_notifications
            if medication_reminders is not None:
                existing_preference.medication_reminders = medication_reminders
            if lab_result_notifications is not None:
                existing_preference.lab_result_notifications = lab_result_notifications

            # Update timestamp
            existing_preference.updated_at = current_time

            # Update the preference in database
            notification_preference_table[preference_id] = existing_preference
        else:
            # Create new preference record with default values for unspecified fields
            preference_id = "PREF-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            new_preference = NotificationPreference(
                preference_id=preference_id,
                patient_id=patient_id,
                appointment_reminders=appointment_reminders if appointment_reminders is not None else True,
                message_notifications=message_notifications if message_notifications is not None else True,
                medication_reminders=medication_reminders if medication_reminders is not None else True,
                lab_result_notifications=lab_result_notifications if lab_result_notifications is not None else True,
                updated_at=current_time
            )

            notification_preference_table[preference_id] = new_preference

        # Save updated preference table back to database
        setattr(db, "notification_preference", notification_preference_table)

        # Handle notification channels if provided
        if notification_channels is not None:
            # Remove existing channels for this preference
            channels_to_remove = []
            for channel_id, channel in notification_channel_table.items():
                if channel.preference_id == preference_id:
                    channels_to_remove.append(channel_id)

            for channel_id in channels_to_remove:
                del notification_channel_table[channel_id]

            # Add new channels
            for channel_type in notification_channels:
                channel_id = "CHAN-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]
                new_channel = NotificationChannel(
                    channel_id=channel_id,
                    preference_id=preference_id,
                    channel_type=channel_type
                )
                notification_channel_table[channel_id] = new_channel

            # Save updated channel table back to database
            setattr(db, "notification_channel", notification_channel_table)

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

        # Return success result
        return {
            "update_status": True,
            "updated_at": updated_at_str
        }

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

        This method updates the appointment's datetime, status, and related metadata.
        It validates the appointment exists, parses the new datetime, and updates
        the appointment record with rescheduling information.
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

        # Retrieve the appointment table from database
        appointment_table = getattr(db, "appointment", None)
        if appointment_table is None:
            raise ValueError("Appointment table not found in database")

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

        # Retrieve the existing appointment
        appointment = appointment_table[appointment_id]

        # Validate that the appointment can be rescheduled (not already completed or cancelled)
        if appointment.status in ["completed", "cancelled"]:
            raise ValueError(f"Cannot reschedule appointment with status '{appointment.status}'")

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

        # Generate a new confirmation number for the rescheduled appointment
        prefix = "CONF-"
        new_confirmation_number = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10].upper()

        # Record the current timestamp for rescheduling
        rescheduled_timestamp = datetime.now()

        # Update the appointment with new information
        appointment.appointment_datetime = new_datetime
        appointment.status = "rescheduled"
        appointment.confirmation_number = new_confirmation_number
        appointment.reschedule_reason = reschedule_reason if reschedule_reason else "No reason provided"
        appointment.rescheduled_at = rescheduled_timestamp

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

        # Return the reschedule status and confirmation details
        return {
            "reschedule_status": True,
            "new_confirmation_number": new_confirmation_number,
            "rescheduled_at": rescheduled_timestamp.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def update_medication_details(
        self,
        medication_id: str,
        dosage: Optional[str] = None,
        frequency: Optional[str] = None,
        instructions: Optional[str] = None
    ) -> dict:
        """
        Update details of an existing medication in patient's medication list.

        Args:
            medication_id: Unique identifier of the medication to update
            dosage: Updated dosage amount and unit (optional)
            frequency: Updated frequency of medication intake (optional)
            instructions: Updated special instructions for taking the medication (optional)

        Returns:
            Dictionary containing:
                - update_status: Boolean indicating if update was successful
                - updated_at: Timestamp when medication was updated in yyyy-mm-dd HH:MM:SS format

        Raises:
            ValueError: If medication_id does not exist or no update fields provided
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the medication table
        medication_table = getattr(db, "medication", None)

        # Validate that medication table exists
        if medication_table is None:
            raise ValueError("Medication table does not exist in database")

        # Check if at least one update field is provided
        if dosage is None and frequency is None and instructions is None:
            raise ValueError("At least one field (dosage, frequency, or instructions) must be provided for update")

        # Validate that medication exists
        if medication_id not in medication_table:
            raise ValueError(f"Medication with ID '{medication_id}' does not exist")

        # Retrieve the existing medication record
        medication_record = medication_table[medication_id]

        # Update the fields that were provided
        if dosage is not None:
            medication_record.dosage = dosage

        if frequency is not None:
            medication_record.frequency = frequency

        if instructions is not None:
            medication_record.instructions = instructions

        # Update the updated_at timestamp
        current_time = datetime.now()
        medication_record.updated_at = current_time

        # Write the updated medication back to the database
        medication_table[medication_id] = medication_record
        setattr(db, "medication", medication_table)

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

        # Return success response
        return {
            "update_status": True,
            "updated_at": updated_at_str
        }

    @is_tool()
    def get_message_thread(self, message_id: str, patient_id: str):
        # Get database instance
        db = self.db

        # Retrieve secure_message table from database
        secure_message_table = getattr(db, "secure_message", None)

        # Validate that secure_message table exists
        if secure_message_table is None:
            raise ValueError("secure_message table does not exist in database")

        # Validate that secure_message table is not empty
        if not secure_message_table:
            raise ValueError(f"Message with ID '{message_id}' not found")

        # Find the message with the given message_id to get the thread_id
        target_message = None
        for msg_id, message in secure_message_table.items():
            if message.message_id == message_id:
                target_message = message
                break

        # Validate that the message exists
        if target_message is None:
            raise ValueError(f"Message with ID '{message_id}' not found")

        # Validate that the patient has access rights to this message
        # Patient must be either the sender or receiver of the message
        if target_message.patient_id != patient_id:
            raise ValueError(f"Patient '{patient_id}' does not have access rights to message '{message_id}'")

        # Get the thread_id from the target message
        thread_id = target_message.thread_id

        # Collect all messages in the same thread
        thread_messages = []
        for msg_id, message in secure_message_table.items():
            # Check if message belongs to the same thread and same patient
            if message.thread_id == thread_id and message.patient_id == patient_id:
                # Build message object with required fields
                message_obj = {
                    "message_id": message.message_id,
                    "sender": message.sender_type,  # "patient" or "provider"
                    "sent_at": message.sent_at.strftime("%Y-%m-%d %H:%M:%S"),  # Format datetime to string
                    "message_body": message.message_body,
                    "subject": message.subject,
                    "priority": message.priority,
                    "is_read": message.is_read,
                    "read_at": message.read_at.strftime("%Y-%m-%d %H:%M:%S") if message.read_at else None
                }
                thread_messages.append(message_obj)

        # Sort messages by sent_at timestamp in chronological order
        thread_messages.sort(key=lambda x: x["sent_at"])

        # Calculate total message count
        message_count = len(thread_messages)

        # Validate that thread has at least one message (should always be true if target_message exists)
        if message_count == 0:
            raise ValueError(f"No messages found in thread '{thread_id}' for patient '{patient_id}'")

        # Return the complete message thread
        return {
            "thread_id": thread_id,
            "messages": thread_messages,
            "message_count": message_count
        }

    @is_tool()
    def remove_medication_from_list(self, medication_id: str, discontinuation_date: str, discontinuation_reason: str = None):
        # Import necessary modules for datetime handling
        from datetime import datetime, date

        # Access the database instance
        db = self.db

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

        # Retrieve the medication table from the database
        medication_table = getattr(db, "medication", None)

        # Check if medication table exists
        if medication_table is None:
            raise ValueError("Medication table does not exist in the database")

        # Check if the medication exists in the table
        if medication_id not in medication_table:
            raise ValueError(f"Medication with ID '{medication_id}' does not exist in patient's medication list")

        # Retrieve the medication record
        medication_record = medication_table[medication_id]

        # Check if the medication is already inactive (already removed)
        if not medication_record.is_active:
            raise ValueError(f"Medication with ID '{medication_id}' has already been removed from the list")

        # Generate the removal timestamp in yyyy-mm-dd HH:MM:SS format
        removed_timestamp = datetime.now()
        removed_at_str = removed_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Update the medication record to mark it as inactive (removed)
        medication_record.is_active = False
        medication_record.discontinuation_date = parsed_discontinuation_date
        medication_record.discontinuation_reason = discontinuation_reason
        medication_record.removed_at = removed_timestamp
        medication_record.updated_at = removed_timestamp

        # Save the updated medication record back to the database
        medication_table[medication_id] = medication_record
        setattr(db, "medication", medication_table)

        # Return the removal status and timestamp
        return {
            "removal_status": True,
            "removed_at": removed_at_str
        }

    @is_tool()
    def calculate_appointment_wait_time(
        self,
        appointment_datetime: str,
        current_time: str,
        provider_average_consultation_minutes: int,
        appointments_ahead_in_queue: int
    ) -> dict:
        """
        Calculate estimated wait time before appointment based on provider schedule and current queue.

        This method computes:
        1. The time until the scheduled appointment
        2. Additional delay caused by appointments ahead in queue
        3. The estimated actual start time considering both factors

        Args:
            appointment_datetime: Scheduled appointment time in "yyyy-mm-dd HH:MM:SS" format
            current_time: Current time in "yyyy-mm-dd HH:MM:SS" format
            provider_average_consultation_minutes: Provider's average consultation duration in minutes
            appointments_ahead_in_queue: Number of appointments scheduled before this one

        Returns:
            dict containing:
                - estimated_wait_minutes: Total estimated wait time in minutes
                - estimated_start_time: Estimated actual start time in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            ValueError: If datetime strings are invalid format, or if calculated wait time is negative,
                       or if parameters have invalid values
        """
        from datetime import datetime, timedelta

        # Validate input parameters
        if provider_average_consultation_minutes < 0:
            raise ValueError("provider_average_consultation_minutes must be non-negative")

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

        # Parse datetime strings with format validation
        try:
            appointment_dt = datetime.strptime(appointment_datetime.strip(), "%Y-%m-%d %H:%M:%S")
        except (ValueError, AttributeError) as e:
            raise ValueError(f"Invalid appointment_datetime format. Expected 'yyyy-mm-dd HH:MM:SS', got: {appointment_datetime}")

        try:
            current_dt = datetime.strptime(current_time.strip(), "%Y-%m-%d %H:%M:%S")
        except (ValueError, AttributeError) as e:
            raise ValueError(f"Invalid current_time format. Expected 'yyyy-mm-dd HH:MM:SS', got: {current_time}")

        # Calculate time until scheduled appointment (in minutes)
        time_until_appointment = (appointment_dt - current_dt).total_seconds() / 60

        # Validate that current_time is not after appointment_datetime
        if time_until_appointment < 0:
            raise ValueError("current_time cannot be after appointment_datetime")

        # Calculate delay caused by appointments ahead in queue
        # Each appointment ahead adds average consultation time to the wait
        queue_delay_minutes = appointments_ahead_in_queue * provider_average_consultation_minutes

        # Total estimated wait time is the maximum of:
        # 1. Time until scheduled appointment + queue delay
        # 2. Just the queue delay (if current time is already past scheduled time)
        # Since we validated current_time <= appointment_datetime, we use option 1
        estimated_wait_minutes = int(time_until_appointment + queue_delay_minutes)

        # Calculate estimated actual start time
        # Start from current time and add the total wait time
        estimated_start_dt = current_dt + timedelta(minutes=estimated_wait_minutes)
        estimated_start_time = estimated_start_dt.strftime("%Y-%m-%d %H:%M:%S")

        return {
            "estimated_wait_minutes": estimated_wait_minutes,
            "estimated_start_time": estimated_start_time
        }

    @is_tool()
    def search_healthcare_providers(
        self,
        specialty: Literal["General Practice", "Cardiology", "Dermatology", "Pediatrics", "Psychiatry", "Orthopedics", "Neurology", "Endocrinology"],
        location: str = None,
        availability_date: str = None,
        insurance_provider: str = None
    ) -> dict:
        """
        Search for available healthcare providers based on specialty, location, and availability criteria.

        This method searches the database for healthcare providers matching the specified criteria.
        It performs fuzzy matching on location fields to improve search robustness.

        Args:
            specialty: Medical specialty of the provider (must be one of the predefined specialties)
            location: Geographic location or zip code (optional, uses fuzzy matching)
            availability_date: Desired appointment date in yyyy-mm-dd format (optional)
            insurance_provider: Insurance provider name (optional, uses fuzzy matching)

        Returns:
            dict: Contains 'providers' (list of matching provider details) and 'total_count' (integer)

        Raises:
            ValueError: If specialty is invalid or date format is incorrect
        """
        from thefuzz import fuzz, process
        from datetime import datetime

        # Validate specialty parameter (already enforced by Literal type, but add runtime check for safety)
        valid_specialties = ["General Practice", "Cardiology", "Dermatology", "Pediatrics", 
                            "Psychiatry", "Orthopedics", "Neurology", "Endocrinology"]
        if specialty not in valid_specialties:
            raise ValueError(f"Invalid specialty. Must be one of: {', '.join(valid_specialties)}")

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

        # Access database
        db = self.db

        # Get all database tables
        provider_specialty_table = getattr(db, "provider_specialty", None)
        healthcare_provider_table = getattr(db, "healthcare_provider", None)
        provider_language_table = getattr(db, "provider_language", None)

        # Check if required tables exist
        if not provider_specialty_table or not healthcare_provider_table:
            return {"providers": [], "total_count": 0}

        # Step 1: Find all provider_ids with matching specialty
        matching_provider_ids = set()
        for specialty_record in provider_specialty_table.values():
            if specialty_record.specialty == specialty:
                matching_provider_ids.add(specialty_record.provider_id)

        # If no providers found with this specialty, return empty result
        if not matching_provider_ids:
            return {"providers": [], "total_count": 0}

        # Step 2: Build detailed provider information
        matching_providers = []

        for provider_id in matching_provider_ids:
            # Get provider details
            provider = healthcare_provider_table.get(provider_id)
            if not provider:
                continue

            # Build provider info dictionary
            provider_info = {
                "provider_id": provider.provider_id,
                "name": provider.full_name,
                "specialty": specialty,
                "credentials": provider.credentials,
                "years_experience": provider.years_experience,
                "average_rating": provider.average_rating,
                "average_consultation_minutes": provider.average_consultation_minutes
            }

            # Add languages if available
            if provider_language_table:
                languages = []
                for lang_record in provider_language_table.values():
                    if lang_record.provider_id == provider_id:
                        languages.append(lang_record.language)
                if languages:
                    provider_info["languages"] = languages

            matching_providers.append(provider_info)

        # Step 3: Apply location filter if specified (using fuzzy matching for natural language text)
        if location:
            # Since location data is not stored in the current schema, we skip this filter
            # In a real implementation, this would filter based on provider location data
            # For now, we'll keep all providers that match specialty
            pass

        # Step 4: Apply availability_date filter if specified
        if availability_date:
            # Since availability data is not stored in the current schema, we skip this filter
            # In a real implementation, this would check provider availability schedules
            # For now, we'll keep all providers that match specialty
            pass

        # Step 5: Apply insurance_provider filter if specified (using fuzzy matching)
        if insurance_provider:
            # Since insurance acceptance data is not stored in the current schema, we skip this filter
            # In a real implementation, this would filter based on accepted insurance providers
            # For now, we'll keep all providers that match specialty
            pass

        # Step 6: Sort providers by rating (descending) for better user experience
        matching_providers.sort(
            key=lambda p: (p.get("average_rating") or 0, p.get("years_experience") or 0),
            reverse=True
        )

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

    @is_tool()
    def get_appointment_details(self, appointment_id: str):
        # Import necessary classes from database schema
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Validate input parameter - appointment_id should not be empty
        if not appointment_id or not isinstance(appointment_id, str):
            raise ValueError("appointment_id must be a non-empty string")

        # Retrieve appointment table from database
        appointment_table = getattr(db, 'appointment', None)
        if appointment_table is None:
            raise ValueError("Appointment table not found in database")

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

        # Retrieve the appointment record
        appointment = appointment_table[appointment_id]

        # Retrieve patient table to verify patient access (pre-condition check)
        patient_table = getattr(db, 'patient', None)
        if patient_table is None or appointment.patient_id not in patient_table:
            raise ValueError(f"Patient with ID '{appointment.patient_id}' not found or does not have access rights")

        # Retrieve healthcare provider table to get provider information
        provider_table = getattr(db, 'healthcare_provider', None)
        if provider_table is None:
            raise ValueError("Healthcare provider table not found in database")

        # Check if the provider exists
        if appointment.provider_id not in provider_table:
            raise ValueError(f"Healthcare provider with ID '{appointment.provider_id}' not found")

        # Retrieve provider details
        provider = provider_table[appointment.provider_id]

        # Format appointment datetime to required string format "yyyy-mm-dd HH:MM:SS"
        # appointment.appointment_datetime is a datetime object from database
        appointment_datetime_str = appointment.appointment_datetime.strftime("%Y-%m-%d %H:%M:%S")

        # Construct and return the complete appointment details dictionary
        # Including all required fields from the returns schema
        return {
            "appointment_id": appointment.appointment_id,
            "patient_id": appointment.patient_id,
            "provider_id": appointment.provider_id,
            "provider_name": provider.full_name,  # Extract provider name from provider record
            "appointment_datetime": appointment_datetime_str,  # Formatted datetime string
            "appointment_type": appointment.appointment_type,
            "status": appointment.status,
            "reason_for_visit": appointment.reason_for_visit
        }

    @is_tool()
    def verify_patient_email(self, patient_id: str, verification_token: str):
        """
        Verify patient email address using verification token sent during registration.

        This method validates the verification token against the patient's account,
        updates the email verification status, and activates the account if successful.
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the patient table from database
        patient_table = getattr(db, "patient", None)
        if patient_table is None:
            raise ValueError("Patient table does not exist in database")

        # Check if patient exists
        if patient_id not in patient_table:
            raise ValueError(f"Patient with ID '{patient_id}' does not exist")

        # Get the patient record
        patient = patient_table[patient_id]

        # Verify that email verification is still pending
        if patient.email_verified:
            raise ValueError(f"Email for patient '{patient_id}' has already been verified")

        # Check account status - should be pending verification
        if patient.account_status not in ["pending_verification", "pending"]:
            raise ValueError(f"Patient account status '{patient.account_status}' does not allow email verification")

        # In a real system, the verification token would be stored in the database
        # or generated using a cryptographic method that can be validated
        # For this implementation, we validate the token format and assume it's correct
        # if it matches the expected pattern
        if not verification_token or not isinstance(verification_token, str):
            raise ValueError("Invalid verification token format")

        # Validate token format (should start with "VT-" based on examples)
        if not verification_token.startswith("VT-"):
            raise ValueError("Verification token format is invalid - must start with 'VT-'")

        # Token validation passed - proceed with email verification
        # Get current timestamp for verification
        verified_timestamp = datetime.now()

        # Update patient record with verification details
        patient.email_verified = True
        patient.account_status = "active"
        patient.verified_at = verified_timestamp
        patient.updated_at = verified_timestamp

        # Save updated patient record back to database
        patient_table[patient_id] = patient
        setattr(db, "patient", patient_table)

        # Format the timestamp as string in required format
        verified_at_str = verified_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return verification result
        return {
            "verification_status": True,
            "account_status": "active",
            "verified_at": verified_at_str
        }

    @is_tool()
    def add_medical_condition(
        self,
        patient_id: str,
        condition_name: str,
        diagnosis_date: str,
        condition_status: Literal["active", "resolved", "chronic", "in_remission"],
        severity: Literal["mild", "moderate", "severe"] = None,
        notes: str = None
    ):
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters are not empty
        if not patient_id or not patient_id.strip():
            raise ValueError("patient_id cannot be empty")
        if not condition_name or not condition_name.strip():
            raise ValueError("condition_name cannot be empty")
        if not diagnosis_date or not diagnosis_date.strip():
            raise ValueError("diagnosis_date cannot be empty")
        if not condition_status or not condition_status.strip():
            raise ValueError("condition_status cannot be empty")

        # Validate and parse diagnosis_date format (yyyy-mm-dd)
        try:
            parsed_diagnosis_date = datetime.strptime(diagnosis_date.strip(), "%Y-%m-%d").date()
        except ValueError:
            raise ValueError(f"diagnosis_date must be in yyyy-mm-dd format, got: {diagnosis_date}")

        # Validate condition_status enum
        valid_statuses = ["active", "resolved", "chronic", "in_remission"]
        if condition_status not in valid_statuses:
            raise ValueError(f"condition_status must be one of {valid_statuses}, got: {condition_status}")

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

        # Generate unique condition_id with COND prefix
        condition_id = "COND-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Access the database
        db = self.db

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

        # Get the medical_condition table
        medical_condition_table = getattr(db, "medical_condition")

        # Create new MedicalCondition instance
        new_condition = MedicalCondition(
            condition_id=condition_id,
            patient_id=patient_id.strip(),
            condition_name=condition_name.strip(),
            diagnosis_date=parsed_diagnosis_date,
            condition_status=condition_status,
            severity=severity,
            notes=notes.strip() if notes else None,
            added_at=added_at
        )

        # Add the new condition to the table (key is condition_id)
        medical_condition_table[condition_id] = new_condition

        # Update the database
        setattr(db, "medical_condition", medical_condition_table)

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

    @is_tool()
    def get_current_medications(self, patient_id: str):
        # Validate input parameter
        if not patient_id or not isinstance(patient_id, str):
            raise ValueError("patient_id must be a non-empty string")

        # Access the database
        db = self.db

        # Get the medication table from database
        medication_table = getattr(db, "medication", None)

        # If medication table doesn't exist or is empty, return empty result
        if medication_table is None or len(medication_table) == 0:
            return {
                "medications": [],
                "total_count": 0
            }

        # Filter medications for the specified patient that are currently active
        current_medications = []

        for medication_id, medication in medication_table.items():
            # Check if medication belongs to the patient and is currently active
            if medication.patient_id == patient_id and medication.is_active:
                # Build medication information dictionary
                medication_info = {
                    "medication_id": medication.medication_id,
                    "medication_name": medication.medication_name,
                    "dosage": medication.dosage,
                    "frequency": medication.frequency
                }

                # Add optional fields if they exist
                if medication.start_date:
                    medication_info["start_date"] = medication.start_date.strftime("%Y-%m-%d")

                if medication.prescribing_provider:
                    medication_info["prescribing_provider"] = medication.prescribing_provider

                if medication.instructions:
                    medication_info["instructions"] = medication.instructions

                current_medications.append(medication_info)

        # Sort medications by start_date (most recent first) if available
        # This provides a consistent ordering for the results
        current_medications.sort(
            key=lambda x: x.get("start_date", ""), 
            reverse=True
        )

        # Return the list of current medications and total count
        return {
            "medications": current_medications,
            "total_count": len(current_medications)
        }

    @is_tool()
    def get_medical_history(self, patient_id: str):
        """
        Retrieve complete medical history including conditions, surgeries, and family history

        Args:
            patient_id: Unique identifier of the patient

        Returns:
            Dictionary containing:
            - conditions: List of medical conditions with details
            - surgeries: List of past surgeries
            - family_history: List of family medical history

        Raises:
            ValueError: If patient_id is invalid or empty
        """
        # Import necessary modules
        from datetime import datetime

        # Validate input parameter
        if not patient_id or not isinstance(patient_id, str) or not patient_id.strip():
            raise ValueError("patient_id must be a non-empty string")

        patient_id = patient_id.strip()

        # Access the database
        db = self.db

        # Get medical_condition table from database
        medical_condition_table = getattr(db, 'medical_condition', None)

        # Initialize result structure
        result = {
            "conditions": [],
            "surgeries": [],
            "family_history": []
        }

        # If medical_condition table doesn't exist or is empty, return empty result
        if medical_condition_table is None or not medical_condition_table:
            return result

        # Iterate through all medical condition records to find matching patient records
        for condition_id, condition_record in medical_condition_table.items():
            # Check if this condition belongs to the specified patient
            if condition_record.patient_id == patient_id:
                # Build condition object with all available information
                condition_obj = {
                    "condition_id": condition_record.condition_id,
                    "condition_name": condition_record.condition_name,
                    "status": condition_record.condition_status
                }

                # Add optional fields if they exist
                if hasattr(condition_record, 'diagnosis_date') and condition_record.diagnosis_date:
                    # Convert date to string format
                    condition_obj["diagnosis_date"] = condition_record.diagnosis_date.strftime("%Y-%m-%d")

                if hasattr(condition_record, 'severity') and condition_record.severity:
                    condition_obj["severity"] = condition_record.severity

                if hasattr(condition_record, 'notes') and condition_record.notes:
                    condition_obj["notes"] = condition_record.notes

                if hasattr(condition_record, 'added_at') and condition_record.added_at:
                    # Convert datetime to string format
                    condition_obj["added_at"] = condition_record.added_at.strftime("%Y-%m-%d %H:%M:%S")

                # Add to conditions list
                result["conditions"].append(condition_obj)

        # Note: Based on the database schema provided, only medical_condition table is available
        # surgeries and family_history are initialized as empty lists as no corresponding tables exist
        # If these tables are added to the database in the future, similar logic should be implemented
        # to populate these fields from their respective tables

        return result

    @is_tool()
    def list_patient_appointments(
        self,
        patient_id: str,
        status_filter: Literal["scheduled", "completed", "cancelled", "no_show", "all"] = "all",
        start_date: str = None,
        end_date: str = None
    ):
        """
        Retrieve list of all appointments for a patient with optional filtering by status and date range.

        Args:
            patient_id: Unique identifier of the patient
            status_filter: Filter appointments by status (scheduled, completed, cancelled, no_show, all)
            start_date: Start date for filtering in yyyy-mm-dd format
            end_date: End date for filtering in yyyy-mm-dd format

        Returns:
            Dictionary containing:
                - appointments: List of patient appointments
                - total_count: Total number of appointments

        Raises:
            ValueError: If patient_id is empty, date format is invalid, or date range is invalid
        """
        from datetime import datetime

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

        # Validate status_filter (safety protection for enum parameter)
        valid_statuses = ["scheduled", "completed", "cancelled", "no_show", "all"]
        if status_filter not in valid_statuses:
            raise ValueError(f"Invalid status_filter: {status_filter}. Must be one of {valid_statuses}")

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

        if start_date:
            try:
                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:
                # Set end_datetime to end of day (23:59:59)
                end_datetime = datetime.strptime(end_date, "%Y-%m-%d")
                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
        if start_datetime and end_datetime and start_datetime > end_datetime:
            raise ValueError("start_date cannot be after end_date")

        # Access the database
        db = self.db
        appointment_table = getattr(db, "appointment", None)

        # If appointment table doesn't exist or is empty, return empty result
        if not appointment_table:
            return {
                "appointments": [],
                "total_count": 0
            }

        # Filter appointments for the specific patient
        filtered_appointments = []

        for appointment_id, appointment in appointment_table.items():
            # Filter by patient_id (exact match for ID field)
            if appointment.patient_id != patient_id:
                continue

            # Filter by status if not "all"
            if status_filter != "all" and appointment.status != status_filter:
                continue

            # Filter by start_date if provided
            if start_datetime and appointment.appointment_datetime < start_datetime:
                continue

            # Filter by end_date if provided
            if end_datetime and appointment.appointment_datetime > end_datetime:
                continue

            # Add matching appointment to results
            filtered_appointments.append(appointment)

        # Sort appointments by datetime (most recent first)
        filtered_appointments.sort(key=lambda x: x.appointment_datetime, reverse=True)

        # Format appointments for return
        formatted_appointments = []
        for appointment in filtered_appointments:
            formatted_appointment = {
                "appointment_id": appointment.appointment_id,
                "provider_id": appointment.provider_id,
                "datetime": appointment.appointment_datetime.strftime("%Y-%m-%d %H:%M:%S"),
                "appointment_type": appointment.appointment_type,
                "reason_for_visit": appointment.reason_for_visit,
                "duration_minutes": appointment.duration_minutes,
                "status": appointment.status,
                "confirmation_number": appointment.confirmation_number
            }
            formatted_appointments.append(formatted_appointment)

        return {
            "appointments": formatted_appointments,
            "total_count": len(formatted_appointments)
        }

    @is_tool()
    def validate_appointment_eligibility(
        self,
        patient_age: int,
        has_active_insurance: bool,
        insurance_covers_telehealth: bool,
        specialty: str,
        appointment_type: Literal["video_consultation", "phone_consultation", "in_person"],
        has_previous_visits: bool = None
    ) -> dict:
        """
        Validate whether a patient is eligible to schedule an appointment based on insurance,
        medical history, and provider requirements.

        This method performs comprehensive eligibility checks including:
        - Age requirements for different specialties
        - Insurance coverage validation
        - Appointment type compatibility with insurance
        - Previous visit requirements for certain appointment types

        Args:
            patient_age: Patient's age in years
            has_active_insurance: Whether patient has active insurance coverage
            insurance_covers_telehealth: Whether insurance covers telehealth services
            specialty: Medical specialty of the provider
            appointment_type: Type of appointment requested (video_consultation, phone_consultation, in_person)
            has_previous_visits: Whether patient has had previous visits with this provider (optional)

        Returns:
            dict: Contains eligibility status, reasons, and required actions
                - is_eligible: Boolean indicating overall eligibility
                - eligibility_reasons: List of reasons affecting eligibility decision
                - required_actions: List of actions needed if not fully eligible

        Raises:
            ValueError: If patient_age is negative or invalid specialty/appointment_type provided
        """
        # Input validation
        if patient_age < 0:
            raise ValueError("Patient age cannot be negative")

        if not isinstance(patient_age, int):
            raise ValueError("Patient age must be an integer")

        # Initialize return structure
        is_eligible = True
        eligibility_reasons = []
        required_actions = []

        # Age requirement validation based on specialty
        # Pediatrics requires patients under 18, other specialties typically require 18+
        if specialty == "Pediatrics":
            if patient_age >= 18:
                is_eligible = False
                eligibility_reasons.append("Patient age exceeds pediatric care limit (must be under 18)")
                required_actions.append("Schedule with adult care provider")
            else:
                eligibility_reasons.append("Age requirement met for pediatric care")
        else:
            # Most specialties require adult patients (18+)
            if patient_age < 18:
                is_eligible = False
                eligibility_reasons.append("Patient must be 18 or older for this specialty")
                required_actions.append("Schedule with pediatric provider or obtain parental consent")
            else:
                eligibility_reasons.append("Age requirement met")

        # Insurance validation
        if not has_active_insurance:
            is_eligible = False
            eligibility_reasons.append("No active insurance coverage")
            required_actions.append("Activate insurance coverage or arrange self-pay")
        else:
            eligibility_reasons.append("Active insurance coverage verified")

        # Telehealth coverage validation for virtual appointments
        if appointment_type in ["video_consultation", "phone_consultation"]:
            if has_active_insurance and not insurance_covers_telehealth:
                is_eligible = False
                eligibility_reasons.append("Insurance does not cover telehealth services")
                required_actions.append("Switch to in-person appointment or arrange self-pay for telehealth")
            elif has_active_insurance and insurance_covers_telehealth:
                eligibility_reasons.append("Insurance covers telehealth services")

        # In-person appointments don't require telehealth coverage check
        if appointment_type == "in_person" and has_active_insurance:
            eligibility_reasons.append("In-person appointment covered by insurance")

        # Previous visit requirement for telehealth appointments
        # Many providers require an initial in-person visit before telehealth
        if appointment_type in ["video_consultation", "phone_consultation"]:
            if has_previous_visits is None:
                # If previous visit status is unknown, flag as potential issue
                eligibility_reasons.append("Previous visit history not verified")
                required_actions.append("Verify previous visit history with provider")
            elif has_previous_visits:
                eligibility_reasons.append("Previous patient relationship exists")
            else:
                # Some specialties may require initial in-person visit
                # This is especially common for specialties requiring physical examination
                if specialty in ["Cardiology", "Orthopedics", "Dermatology"]:
                    is_eligible = False
                    eligibility_reasons.append("Initial in-person visit required for this specialty")
                    required_actions.append("Schedule initial in-person consultation before telehealth visits")
                else:
                    # For other specialties, new patients may be accepted for telehealth
                    eligibility_reasons.append("New patient telehealth visit accepted for this specialty")

        # Specialty-specific requirements
        # Some specialties may have additional requirements
        if specialty == "Psychiatry":
            # Psychiatry often allows new patient telehealth visits
            if appointment_type in ["video_consultation", "phone_consultation"] and not has_previous_visits:
                eligibility_reasons.append("New patient telehealth visits accepted for psychiatry")

        # Emergency care considerations
        # Certain conditions should not be handled via telehealth
        if specialty == "Cardiology" and appointment_type in ["video_consultation", "phone_consultation"]:
            eligibility_reasons.append("Note: Acute cardiac symptoms require in-person or emergency care")

        return {
            "is_eligible": is_eligible,
            "eligibility_reasons": eligibility_reasons,
            "required_actions": required_actions
        }

    @is_tool()
    def list_patient_messages(self, patient_id: str, status_filter: Literal["unread", "read", "all"] = "all", provider_id: str = None):
        """
        Retrieve list of all message threads for a patient with optional filtering

        Args:
            patient_id: Unique identifier of the patient
            status_filter: Filter messages by read status (unread/read/all)
            provider_id: Optional filter for messages from a specific provider

        Returns:
            Dictionary containing:
            - message_threads: List of message thread summaries
            - unread_count: Number of unread messages
            - total_threads: Total number of message threads

        Raises:
            ValueError: If patient_id is empty or invalid parameters provided
        """
        from datetime import datetime

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

        # Validate status_filter enum value (safety protection)
        valid_statuses = ["unread", "read", "all"]
        if status_filter not in valid_statuses:
            raise ValueError(f"status_filter must be one of {valid_statuses}, got: {status_filter}")

        # Access database
        db = self.db
        secure_message_table = getattr(db, "secure_message", None)

        # Handle empty database case
        if secure_message_table is None or not secure_message_table:
            return {
                "message_threads": [],
                "unread_count": 0,
                "total_threads": 0
            }

        # Filter messages by patient_id
        patient_messages = [
            msg for msg in secure_message_table.values()
            if msg.patient_id == patient_id
        ]

        # Apply provider_id filter if specified
        if provider_id is not None and provider_id.strip():
            patient_messages = [
                msg for msg in patient_messages
                if msg.provider_id == provider_id
            ]

        # Apply status filter
        if status_filter == "unread":
            filtered_messages = [msg for msg in patient_messages if not msg.is_read]
        elif status_filter == "read":
            filtered_messages = [msg for msg in patient_messages if msg.is_read]
        else:  # status_filter == "all"
            filtered_messages = patient_messages

        # Group messages by thread_id to create thread summaries
        thread_dict = {}
        for msg in filtered_messages:
            thread_id = msg.thread_id
            if thread_id not in thread_dict:
                thread_dict[thread_id] = {
                    "messages": [],
                    "subject": msg.subject
                }
            thread_dict[thread_id]["messages"].append(msg)

        # Build message thread summaries
        message_threads = []
        for thread_id, thread_data in thread_dict.items():
            # Find the most recent message in this thread
            latest_message = max(thread_data["messages"], key=lambda m: m.sent_at)

            # Format the thread summary
            thread_summary = {
                "thread_id": thread_id,
                "subject": thread_data["subject"],
                "last_message_at": latest_message.sent_at.strftime("%Y-%m-%d %H:%M:%S")
            }
            message_threads.append(thread_summary)

        # Sort threads by last_message_at in descending order (most recent first)
        message_threads.sort(
            key=lambda t: datetime.strptime(t["last_message_at"], "%Y-%m-%d %H:%M:%S"),
            reverse=True
        )

        # Calculate unread count from all patient messages (not just filtered ones)
        unread_count = sum(1 for msg in patient_messages if not msg.is_read)

        # Total threads count
        total_threads = len(message_threads)

        return {
            "message_threads": message_threads,
            "unread_count": unread_count,
            "total_threads": total_threads
        }

    @is_tool()
    def get_provider_details(self, provider_id: str):
        # Get database instance
        db = self.db

        # Retrieve healthcare_provider table
        healthcare_provider_table = getattr(db, 'healthcare_provider', None)
        if healthcare_provider_table is None:
            raise ValueError("healthcare_provider table not found in database")

        # Check if provider exists
        provider = healthcare_provider_table.get(provider_id)
        if provider is None:
            raise ValueError(f"Provider with ID '{provider_id}' does not exist in the system")

        # Retrieve provider_specialty table to get specialties
        provider_specialty_table = getattr(db, 'provider_specialty', None)
        specialties = []
        if provider_specialty_table is not None:
            # Filter specialties by provider_id
            for specialty_record in provider_specialty_table.values():
                if specialty_record.provider_id == provider_id:
                    specialties.append(specialty_record.specialty)

        # Retrieve provider_language table to get languages spoken
        provider_language_table = getattr(db, 'provider_language', None)
        languages_spoken = []
        if provider_language_table is not None:
            # Filter languages by provider_id
            for language_record in provider_language_table.values():
                if language_record.provider_id == provider_id:
                    languages_spoken.append(language_record.language)

        # Construct and return provider details
        provider_details = {
            "provider_id": provider.provider_id,
            "full_name": provider.full_name,
            "specialties": specialties,
            "credentials": provider.credentials if provider.credentials else "",
            "years_experience": provider.years_experience if provider.years_experience is not None else 0,
            "average_rating": provider.average_rating if provider.average_rating is not None else 0.0,
            "languages_spoken": languages_spoken
        }

        return provider_details

    @is_tool()
    def set_medication_reminder(
        self,
        medication_id: str,
        reminder_times: list,
        notification_method: Literal["push_notification", "sms", "email", "all"],
        reminder_enabled: bool = True
    ):
        """
        Set up a reminder notification for taking a specific medication.

        This method creates a new medication reminder with specified times and notification method.
        It generates a unique reminder_id and stores the reminder configuration in the database.

        Args:
            medication_id: Unique identifier of the medication
            reminder_times: List of times to send reminders in HH:MM format (e.g., ["08:00", "20:00"])
            notification_method: Method for sending reminders (must be one of the enum values)
            reminder_enabled: Whether the reminder is enabled (default: True)

        Returns:
            dict: Contains reminder_id, setup_status, and created_at timestamp

        Raises:
            ValueError: If medication_id is empty, reminder_times is empty/invalid,
                       or notification_method is not in allowed values
        """
        import secrets
        import hashlib
        from datetime import datetime, time

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

        # Validate reminder_times
        if not reminder_times or not isinstance(reminder_times, list):
            raise ValueError("reminder_times must be a non-empty list")

        if len(reminder_times) == 0:
            raise ValueError("reminder_times must contain at least one time")

        # Validate each time format in reminder_times
        parsed_times = []
        for time_str in reminder_times:
            if not isinstance(time_str, str):
                raise ValueError(f"Each reminder time must be a string, got {type(time_str)}")

            try:
                # Parse time string in HH:MM format
                time_obj = datetime.strptime(time_str.strip(), "%H:%M").time()
                parsed_times.append(time_obj)
            except ValueError:
                raise ValueError(f"Invalid time format '{time_str}'. Expected HH:MM format (e.g., '08:00')")

        # Validate notification_method enum
        allowed_methods = ["push_notification", "sms", "email", "all"]
        if notification_method not in allowed_methods:
            raise ValueError(
                f"notification_method must be one of {allowed_methods}, got '{notification_method}'"
            )

        # Access database
        db = self.db

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

        # Create timestamp for created_at
        created_at = datetime.now()

        # Create MedicationReminder object
        medication_reminder = MedicationReminder(
            reminder_id=reminder_id,
            medication_id=medication_id,
            notification_method=notification_method,
            reminder_enabled=reminder_enabled,
            created_at=created_at
        )

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

        # Add new reminder to table
        medication_reminder_table[reminder_id] = medication_reminder
        setattr(db, "medication_reminder", medication_reminder_table)

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

        # Create ReminderTime entries for each reminder time
        for time_obj in parsed_times:
            # Generate unique reminder_time_id
            time_prefix = "RTIME-"
            reminder_time_id = time_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Create ReminderTime object
            reminder_time_entry = ReminderTime(
                reminder_time_id=reminder_time_id,
                reminder_id=reminder_id,
                reminder_time=time_obj
            )

            # Add to reminder_time table
            reminder_time_table[reminder_time_id] = reminder_time_entry

        # Update reminder_time table in database
        setattr(db, "reminder_time", reminder_time_table)

        # Return success response with formatted timestamp
        return {
            "reminder_id": reminder_id,
            "setup_status": True,
            "created_at": created_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def mark_message_as_read(self, message_id: str, patient_id: str):
        """
        Mark a message or message thread as read by the patient.

        This method updates the read status of a specific message, setting is_read to True
        and recording the timestamp when it was marked as read.

        Args:
            message_id: Unique identifier of the message to mark as read
            patient_id: Unique identifier of the patient marking the message as read

        Returns:
            dict: Contains update_status (bool) and read_at (str in yyyy-mm-dd HH:MM:SS format)

        Raises:
            ValueError: If message doesn't exist or doesn't belong to the patient
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the secure_message table
        secure_message_table = getattr(db, "secure_message", None)

        # Validate that the table exists
        if secure_message_table is None:
            raise ValueError(f"Message with ID '{message_id}' does not exist")

        # Retrieve the message by message_id
        message = secure_message_table.get(message_id)

        # Validate that the message exists
        if message is None:
            raise ValueError(f"Message with ID '{message_id}' does not exist")

        # Validate that the message belongs to the patient
        if message.patient_id != patient_id:
            raise ValueError(f"Message with ID '{message_id}' does not belong to patient '{patient_id}'")

        # Record the current timestamp when marking as read
        read_timestamp = datetime.now()

        # Update the message's read status and read timestamp
        message.is_read = True
        message.read_at = read_timestamp

        # Save the updated message back to the database
        secure_message_table[message_id] = message
        setattr(db, "secure_message", secure_message_table)

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

        # Return the update status and formatted timestamp
        return {
            "update_status": True,
            "read_at": read_at_str
        }

    @is_tool()
    def record_vital_signs(
        self,
        patient_id: str,
        measurement_datetime: str,
        blood_pressure_systolic: int = None,
        blood_pressure_diastolic: int = None,
        heart_rate: int = None,
        temperature: float = None,
        oxygen_saturation: int = None,
        weight: float = None
    ):
        """
        Record patient's vital signs measurements from home monitoring.

        This method creates a new vital signs record in the database with the provided measurements.
        At least patient_id and measurement_datetime are required, while other vital sign measurements
        are optional and can be recorded as available.
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

        # Parse and validate measurement_datetime format (yyyy-mm-dd HH:MM:SS)
        try:
            parsed_measurement_datetime = datetime.strptime(measurement_datetime, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError("measurement_datetime must be in 'yyyy-mm-dd HH:MM:SS' format (e.g., '2024-01-16 08:00:00')")

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

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

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

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

        if oxygen_saturation is not None:
            if not isinstance(oxygen_saturation, int) or oxygen_saturation < 0 or oxygen_saturation > 100:
                raise ValueError("oxygen_saturation must be an integer between 0 and 100")

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

        # Generate unique vital signs ID
        vital_signs_id = "VS-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Access the database
        db = self.db

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

        # Create new VitalSign record
        new_vital_sign = VitalSign(
            vital_signs_id=vital_signs_id,
            patient_id=patient_id,
            measurement_datetime=parsed_measurement_datetime,
            blood_pressure_systolic=blood_pressure_systolic,
            blood_pressure_diastolic=blood_pressure_diastolic,
            heart_rate=heart_rate,
            temperature=temperature,
            oxygen_saturation=oxygen_saturation,
            weight=weight,
            recorded_at=recorded_at
        )

        # Add the new record to the table
        vital_sign_table[vital_signs_id] = new_vital_sign

        # Save the updated table back to database
        setattr(db, "vital_sign", vital_sign_table)

        # Return the result with formatted datetime strings
        return {
            "vital_signs_id": vital_signs_id,
            "recorded_at": recorded_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def upload_medical_document(
        self,
        patient_id: str,
        document_type: Literal["lab_result", "prescription", "imaging_report", "insurance_card", "medical_history", "vaccination_record"],
        file_name: str,
        file_size_bytes: int,
        file_format: Literal["pdf", "jpg", "png", "docx"],
        description: str = None
    ):
        from datetime import datetime
        import secrets
        import hashlib

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

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

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

        # Validate document_type enum
        valid_document_types = ["lab_result", "prescription", "imaging_report", "insurance_card", "medical_history", "vaccination_record"]
        if document_type not in valid_document_types:
            raise ValueError(f"document_type must be one of {valid_document_types}, got: {document_type}")

        # Validate file_format enum
        valid_file_formats = ["pdf", "jpg", "png", "docx"]
        if file_format not in valid_file_formats:
            raise ValueError(f"file_format must be one of {valid_file_formats}, got: {file_format}")

        # Generate unique document ID using secure random hash
        prefix = "DOC-"
        document_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for upload time
        uploaded_at = datetime.now()

        # Construct file storage path (simulated path based on patient and document type)
        file_path = f"/medical_documents/{patient_id}/{document_type}/{document_id}_{file_name}"

        # Create new MedicalDocument instance (MedicalDocument class is imported at file header)
        new_document = MedicalDocument(
            document_id=document_id,
            patient_id=patient_id,
            document_type=document_type,
            file_name=file_name,
            file_size_bytes=file_size_bytes,
            file_format=file_format,
            description=description,
            file_path=file_path,
            upload_status="success",
            uploaded_at=uploaded_at,
            deleted_at=None,
            deletion_reason=None
        )

        # Access database and store the document
        db = self.db

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

        # Get the medical_document table
        medical_document_table = getattr(db, "medical_document")

        # Store the new document in the table using document_id as key
        medical_document_table[document_id] = new_document

        # Update the database with modified table
        setattr(db, "medical_document", medical_document_table)

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

        # Return success response with document details
        return {
            "document_id": document_id,
            "upload_status": "success",
            "uploaded_at": uploaded_at_str
        }

    @is_tool()
    def download_medical_document(self, document_id: str, patient_id: str):
        """
        Download a medical document from patient's health records.

        This method validates the document existence and patient access rights,
        then generates a secure temporary download URL with expiration time.

        Args:
            document_id: Unique identifier of the document to download
            patient_id: Unique identifier of the patient requesting the download

        Returns:
            dict: Contains download_url, file_name, file_size_bytes, and expires_at

        Raises:
            ValueError: If document doesn't exist, patient doesn't have access,
                       or document has been deleted
        """
        import secrets
        import hashlib
        from datetime import datetime, timedelta

        # Access the database
        db = self.db

        # Retrieve medical_document table
        medical_document_table = getattr(db, "medical_document", None)

        # Validate that the table exists
        if medical_document_table is None:
            raise ValueError("Medical document table not found in database")

        # Check if document exists in the database
        document = medical_document_table.get(document_id)
        if document is None:
            raise ValueError(f"Document with ID '{document_id}' does not exist")

        # Verify that the document belongs to the specified patient (access rights check)
        if document.patient_id != patient_id:
            raise ValueError(
                f"Patient '{patient_id}' does not have access rights to document '{document_id}'"
            )

        # Check if the document has been deleted
        if document.deleted_at is not None:
            raise ValueError(
                f"Document '{document_id}' has been deleted and cannot be downloaded. "
                f"Deletion reason: {document.deletion_reason or 'Not specified'}"
            )

        # Verify upload status is successful
        if document.upload_status != "success":
            raise ValueError(
                f"Document '{document_id}' upload status is '{document.upload_status}', "
                f"cannot download incomplete or failed uploads"
            )

        # Generate a secure token for the download URL
        # Using secrets for cryptographically strong random token
        secure_token = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:32]

        # Construct the secure download URL
        # The URL includes the document ID and a temporary access token
        base_url = "https://telehealth.example.com/documents"
        download_url = f"{base_url}/{document_id}?token={secure_token}"

        # Set expiration time for the download URL (1 hour from now)
        # This ensures temporary access and enhances security
        expires_at = datetime.now() + timedelta(hours=1)
        expires_at_str = expires_at.strftime("%Y-%m-%d %H:%M:%S")

        # Return the download information
        return {
            "download_url": download_url,
            "file_name": document.file_name,
            "file_size_bytes": document.file_size_bytes,
            "expires_at": expires_at_str
        }

    @is_tool()
    def request_prescription_refill(
        self,
        patient_id: str,
        prescription_id: str,
        pharmacy_id: str,
        urgency_level: Literal["routine", "urgent", "emergency"] = "routine",
        additional_notes: str = None
    ):
        # Import necessary modules for ID generation and datetime handling
        import secrets
        import hashlib
        from datetime import datetime

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

        # Validate urgency_level is within allowed enum values
        # Add safety protection for enum parameter
        allowed_urgency_levels = ["routine", "urgent", "emergency"]
        if urgency_level not in allowed_urgency_levels:
            raise ValueError(
                f"urgency_level must be one of {allowed_urgency_levels}, got '{urgency_level}'"
            )

        # Access the database instance
        db = self.db

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

        # Generate unique refill request ID using secure hash
        prefix = "REFILL-"
        refill_request_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

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

        # Create new prescription refill request object
        # Set default status to "pending_approval" as per post_condition
        new_refill_request = PrescriptionRefillRequest(
            refill_request_id=refill_request_id,
            patient_id=patient_id,
            prescription_id=prescription_id,
            pharmacy_id=pharmacy_id,
            urgency_level=urgency_level,
            additional_notes=additional_notes,
            request_status="pending_approval",
            submitted_at=submitted_at
        )

        # Add the new refill request to the database
        refill_requests[refill_request_id] = new_refill_request
        setattr(db, "prescription_refill_request", refill_requests)

        # Return the required information in the specified format
        return {
            "refill_request_id": refill_request_id,
            "request_status": "pending_approval",
            "submitted_at": submitted_at.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def get_vital_signs_history(
        self,
        patient_id: str,
        start_date: Optional[str] = None,
        end_date: Optional[str] = None,
        measurement_type: Literal["blood_pressure", "heart_rate", "temperature", "oxygen_saturation", "weight", "all"] = "all"
    ) -> dict:
        """
        Retrieve historical vital signs data for a patient with date range filtering.

        Args:
            patient_id: Unique identifier of the patient
            start_date: Start date for filtering in yyyy-mm-dd format (optional)
            end_date: End date for filtering in yyyy-mm-dd format (optional)
            measurement_type: Specific vital sign to retrieve (default: "all")

        Returns:
            Dictionary containing:
            - vital_signs_records: List of vital signs measurements
            - record_count: Total number of records

        Raises:
            ValueError: If patient_id is empty, date format is invalid, or date range is invalid
        """
        from datetime import datetime

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

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

        if start_date:
            try:
                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:
                end_datetime = datetime.strptime(end_date, "%Y-%m-%d")
                # Set end_datetime to end of day (23:59:59) to include all records on that day
                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
        if start_datetime and end_datetime and start_datetime > end_datetime:
            raise ValueError("start_date cannot be after end_date")

        # Access database
        db = self.db
        vital_sign_table = getattr(db, "vital_sign", None)

        # If table doesn't exist or is empty, return empty result
        if not vital_sign_table:
            return {
                "vital_signs_records": [],
                "record_count": 0
            }

        # Filter vital signs records by patient_id and date range
        filtered_records = []

        for vital_signs_id, vital_sign in vital_sign_table.items():
            # Filter by patient_id
            if vital_sign.patient_id != patient_id:
                continue

            # Filter by date range
            measurement_dt = vital_sign.measurement_datetime

            if start_datetime and measurement_dt < start_datetime:
                continue

            if end_datetime and measurement_dt > end_datetime:
                continue

            # Build record based on measurement_type
            record = {
                "measurement_datetime": measurement_dt.strftime("%Y-%m-%d %H:%M:%S")
            }

            # Add requested measurements based on measurement_type
            if measurement_type == "all" or measurement_type == "blood_pressure":
                if vital_sign.blood_pressure_systolic is not None and vital_sign.blood_pressure_diastolic is not None:
                    record["blood_pressure"] = f"{vital_sign.blood_pressure_systolic}/{vital_sign.blood_pressure_diastolic}"

            if measurement_type == "all" or measurement_type == "heart_rate":
                if vital_sign.heart_rate is not None:
                    record["heart_rate"] = vital_sign.heart_rate

            if measurement_type == "all" or measurement_type == "temperature":
                if vital_sign.temperature is not None:
                    record["temperature"] = vital_sign.temperature

            if measurement_type == "all" or measurement_type == "oxygen_saturation":
                if vital_sign.oxygen_saturation is not None:
                    record["oxygen_saturation"] = vital_sign.oxygen_saturation

            if measurement_type == "all" or measurement_type == "weight":
                if vital_sign.weight is not None:
                    record["weight"] = vital_sign.weight

            # Only include records that have at least one measurement (besides datetime)
            if len(record) > 1:
                filtered_records.append(record)

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

        return {
            "vital_signs_records": filtered_records,
            "record_count": len(filtered_records)
        }

    @is_tool()
    def add_insurance_information(
        self,
        patient_id: str,
        insurance_provider: str,
        policy_number: str,
        policy_holder_name: str,
        relationship_to_patient: Literal["self", "spouse", "parent", "child", "other"],
        group_number: str = None,
        coverage_start_date: str = None,
        coverage_end_date: str = None
    ):
        from datetime import datetime
        import secrets
        import hashlib

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

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

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

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

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

        # Validate relationship_to_patient enum value
        valid_relationships = ["self", "spouse", "parent", "child", "other"]
        if relationship_to_patient not in valid_relationships:
            raise ValueError(f"relationship_to_patient must be one of {valid_relationships}, got '{relationship_to_patient}'")

        # Parse and validate coverage dates if provided
        coverage_start_dt = None
        coverage_end_dt = None

        if coverage_start_date is not None:
            if not isinstance(coverage_start_date, str) or not coverage_start_date.strip():
                raise ValueError("coverage_start_date must be a non-empty string in yyyy-mm-dd format")
            try:
                coverage_start_dt = datetime.strptime(coverage_start_date.strip(), "%Y-%m-%d")
            except ValueError:
                raise ValueError(f"coverage_start_date must be in yyyy-mm-dd format, got '{coverage_start_date}'")

        if coverage_end_date is not None:
            if not isinstance(coverage_end_date, str) or not coverage_end_date.strip():
                raise ValueError("coverage_end_date must be a non-empty string in yyyy-mm-dd format")
            try:
                coverage_end_dt = datetime.strptime(coverage_end_date.strip(), "%Y-%m-%d")
            except ValueError:
                raise ValueError(f"coverage_end_date must be in yyyy-mm-dd format, got '{coverage_end_date}'")

        # Validate date logic: end date should not be before start date
        if coverage_start_dt and coverage_end_dt and coverage_end_dt < coverage_start_dt:
            raise ValueError("coverage_end_date cannot be before coverage_start_date")

        # Generate unique insurance ID using timestamp and random component for better uniqueness
        insurance_id = "INS-" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        added_at = datetime.now()

        # Access the database
        db = self.db

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

        # Create new insurance record
        new_insurance = Insurance(
            insurance_id=insurance_id,
            patient_id=patient_id.strip(),
            insurance_provider=insurance_provider.strip(),
            policy_number=policy_number.strip(),
            group_number=group_number.strip() if group_number else None,
            policy_holder_name=policy_holder_name.strip(),
            relationship_to_patient=relationship_to_patient,
            coverage_start_date=coverage_start_dt,
            coverage_end_date=coverage_end_dt,
            verification_status="pending_verification",
            added_at=added_at
        )

        # Get current insurance data
        insurance_data = getattr(db, "insurance")

        # Add new insurance record to the database
        insurance_data[insurance_id] = new_insurance

        # Update the database
        setattr(db, "insurance", insurance_data)

        # Return the result in the required format with timestamp formatted as string
        return {
            "insurance_id": insurance_id,
            "verification_status": "pending_verification",
            "added_at": added_at.strftime("%Y-%m-%d %H:%M:%S")
        }
