"""
Database schemas for healthcare_telemedicine_Telehealth_Patient_Portal

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

Generated by ClassDBAgent.
"""

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

@with_instance_key("patient_id")
class Patient(BaseModel, ThreadSafeBase["Patient"]):      
    patient_id: str = Field(..., description="Unique identifier for the patient account")
    first_name: str = Field(..., description="Patient's first name")
    last_name: str = Field(..., description="Patient's last name")
    email: str = Field(..., description="Patient's email address for account login")
    phone_number: str = Field(..., description="Patient's primary contact phone number")
    date_of_birth: date = Field(..., description="Patient's date of birth")
    password_hash: str = Field(..., description="Hashed account password")
    address: Optional[str] = Field(None, description="Patient's residential address")
    emergency_contact_name: Optional[str] = Field(None, description="Name of emergency contact person")
    emergency_contact_phone: Optional[str] = Field(None, description="Phone number of emergency contact")
    preferred_language: Optional[Literal["English", "Spanish", "French", "Chinese", "German"]] = Field(
        None, description="Patient's preferred language for communication"
    )
    account_status: str = Field("pending_verification", description="Current status of the patient account")
    email_verified: bool = Field(False, description="Whether email has been verified")
    two_factor_enabled: bool = Field(False, description="Whether two-factor authentication is enabled")
    two_factor_method: Optional[Literal["sms", "email", "authenticator_app"]] = Field(
        None, description="Method for two-factor authentication"
    )
    created_at: datetime = Field(..., description="Timestamp when the account was created")
    updated_at: Optional[datetime] = Field(None, description="Timestamp when profile was last updated")
    verified_at: Optional[datetime] = Field(None, description="Timestamp when email was verified")

@with_instance_key("provider_id")
class HealthcareProvider(BaseModel, ThreadSafeBase["HealthcareProvider"]):
    provider_id: str = Field(..., description="Unique identifier of the healthcare provider")
    full_name: str = Field(..., description="Provider's full name with title")
    credentials: Optional[str] = Field(default=None, description="Professional credentials and certifications")
    years_experience: Optional[int] = Field(default=None, description="Number of years in practice")
    average_rating: Optional[float] = Field(default=None, description="Average patient rating out of 5")
    average_consultation_minutes: Optional[int] = Field(default=30, description="Average consultation duration in minutes")
    created_at: datetime = Field(..., description="Timestamp when provider was added to system")

@with_instance_key("specialty_id")
class ProviderSpecialty(BaseModel, ThreadSafeBase["ProviderSpecialty"]):
    specialty_id: str = Field(..., description="Unique identifier for the specialty record")
    provider_id: str = Field(..., description="Reference to healthcare provider")
    specialty: Literal[
        "General Practice", 
        "Cardiology", 
        "Dermatology", 
        "Pediatrics", 
        "Psychiatry", 
        "Orthopedics", 
        "Neurology", 
        "Endocrinology"
    ] = Field(..., description="Medical specialty")

@with_instance_key("language_id")
class ProviderLanguage(BaseModel, ThreadSafeBase["ProviderLanguage"]):
    language_id: str = Field(..., description="Unique identifier for the language record")
    provider_id: str = Field(..., description="Reference to healthcare provider")
    language: str = Field(..., description="Language spoken by provider")

@with_instance_key("appointment_id")
class Appointment(BaseModel, ThreadSafeBase["Appointment"]):
    appointment_id: str = Field(..., description="Unique identifier for the appointment")
    patient_id: str = Field(..., description="Reference to patient")
    provider_id: str = Field(..., description="Reference to healthcare provider")
    appointment_datetime: datetime = Field(..., description="Scheduled date and time of appointment")
    appointment_type: Literal["video_consultation", "phone_consultation", "in_person"] = Field(..., description="Type of appointment")
    reason_for_visit: str = Field(..., description="Brief description of the reason for the appointment")
    duration_minutes: Optional[int] = Field(30, description="Expected duration of appointment in minutes")
    status: Literal["scheduled", "completed", "cancelled", "no_show", "rescheduled"] = Field("scheduled", description="Current status of the appointment")
    confirmation_number: Optional[str] = Field(None, description="Appointment confirmation number")
    cancellation_reason: Optional[str] = Field(None, description="Reason for cancelling the appointment")
    cancelled_by: Optional[Literal["patient", "provider", "system"]] = Field(None, description="Who initiated the cancellation")
    reschedule_reason: Optional[str] = Field(None, description="Reason for rescheduling")
    created_at: datetime = Field(..., description="Timestamp when appointment was created")
    cancelled_at: Optional[datetime] = Field(None, description="Timestamp when appointment was cancelled")
    rescheduled_at: Optional[datetime] = Field(None, description="Timestamp when appointment was rescheduled")

@with_instance_key("document_id")
class MedicalDocument(BaseModel, ThreadSafeBase["MedicalDocument"]):
    document_id: str = Field(..., description="Unique identifier for the document")
    patient_id: str = Field(..., description="Reference to patient")
    document_type: Literal[
        "lab_result", "prescription", "imaging_report", "insurance_card", "medical_history", "vaccination_record"
    ] = Field(..., description="Type of medical document")
    file_name: str = Field(..., description="Name of the file")
    file_size_bytes: int = Field(..., description="Size of the file in bytes")
    file_format: Literal["pdf", "jpg", "png", "docx"] = Field(..., description="Format of the file")
    description: Optional[str] = Field(None, description="Brief description of the document")
    file_path: str = Field(..., description="Storage path of the document")
    upload_status: str = Field(default="success", description="Status of the upload operation")
    uploaded_at: datetime = Field(..., description="Timestamp when document was uploaded")
    deleted_at: Optional[datetime] = Field(None, description="Timestamp when document was deleted")
    deletion_reason: Optional[str] = Field(None, description="Reason for deleting the document")

    @validator("uploaded_at", "deleted_at", pre=True)
    def strip_and_parse_datetime(cls, v):
        if v is None:
            return None
        if isinstance(v, str):
            v = v.strip()
            # try parsing with common formats
            try:
                return datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                # fallback to pydantic parse (handles ISO format etc.)
                return datetime.fromisoformat(v)
        return v

@with_instance_key("medication_id")
class Medication(BaseModel, ThreadSafeBase["Medication"]):      
    medication_id: str = Field(..., description="Unique identifier for the medication entry")
    patient_id: str = Field(..., description="Reference to patient")
    medication_name: str = Field(..., description="Name of the medication")
    dosage: str = Field(..., description="Dosage amount and unit")
    frequency: str = Field(..., description="How often the medication should be taken")
    start_date: date = Field(..., description="Date when medication was started")
    prescribing_provider: Optional[str] = Field(default=None, description="Name of the provider who prescribed the medication")
    instructions: Optional[str] = Field(default=None, description="Special instructions for taking the medication")
    is_active: bool = Field(default=True, description="Whether medication is currently active")
    discontinuation_date: Optional[date] = Field(default=None, description="Date when medication was discontinued")
    discontinuation_reason: Optional[str] = Field(default=None, description="Reason for discontinuing the medication")
    added_at: datetime = Field(..., description="Timestamp when medication was added")
    updated_at: Optional[datetime] = Field(default=None, description="Timestamp when medication was updated")
    removed_at: Optional[datetime] = Field(default=None, description="Timestamp when medication was removed")

@with_instance_key("reminder_id")
class MedicationReminder(BaseModel, ThreadSafeBase["MedicationReminder"]):
    reminder_id: str = Field(..., description="Unique identifier for the medication reminder")
    medication_id: str = Field(..., description="Reference to medication")
    notification_method: Literal["push_notification", "sms", "email", "all"] = Field(
        ..., description="Method for sending reminders"
    )
    reminder_enabled: bool = Field(
        default=True, description="Whether the reminder is enabled"
    )
    created_at: datetime = Field(
        ..., description="Timestamp when reminder was created"
    )

@with_instance_key("reminder_time_id")
class ReminderTime(BaseModel, ThreadSafeBase["ReminderTime"]):
    reminder_time_id: str = Field(..., description="Unique identifier for the reminder time")
    reminder_id: str = Field(..., description="Reference to medication reminder")
    reminder_time: time = Field(..., description="Time to send reminder in HH:MM format")

@with_instance_key("prescription_id")
class Prescription(BaseModel, ThreadSafeBase["Prescription"]):
    prescription_id: str = Field(..., description="Unique identifier of the prescription")
    patient_id: str = Field(..., description="Reference to patient")
    provider_id: str = Field(..., description="Reference to prescribing provider")
    medication_name: str = Field(..., description="Name of prescribed medication")
    prescribed_date: date = Field(..., description="Date when prescription was written")
    status: Literal["active", "completed", "cancelled"] = Field(..., description="Current status of the prescription")
    refills_remaining: Optional[int] = Field(default=None, description="Number of refills remaining")
    created_at: datetime = Field(..., description="Timestamp when prescription was created")

@with_instance_key("refill_request_id")
class PrescriptionRefillRequest(BaseModel, ThreadSafeBase["PrescriptionRefillRequest"]):
    refill_request_id: str = Field(..., description="Unique identifier for the refill request")
    patient_id: str = Field(..., description="Reference to patient")
    prescription_id: str = Field(..., description="Reference to prescription")
    pharmacy_id: str = Field(..., description="Reference to preferred pharmacy")
    urgency_level: Optional[Literal["routine", "urgent", "emergency"]] = Field(
        default="routine",
        description="Urgency level of the refill request"
    )
    additional_notes: Optional[str] = Field(
        default=None,
        description="Additional notes or comments for the refill request"
    )
    request_status: str = Field(
        default="pending_approval",
        description="Status of the refill request"
    )
    submitted_at: datetime = Field(
        ...,
        description="Timestamp when request was submitted"
    )

@with_instance_key("pharmacy_id")
class Pharmacy(BaseModel, ThreadSafeBase["Pharmacy"]):
    pharmacy_id: str = Field(..., description="Unique identifier for the pharmacy entry")
    patient_id: str = Field(..., description="Reference to patient")
    pharmacy_name: str = Field(..., description="Name of the pharmacy")
    pharmacy_address: str = Field(..., description="Full address of the pharmacy")
    pharmacy_phone: str = Field(..., description="Phone number of the pharmacy")
    is_primary: bool = Field(default=False, description="Whether this is the primary pharmacy")
    added_at: datetime = Field(..., description="Timestamp when pharmacy was added")

@with_instance_key("message_id")
class SecureMessage(BaseModel, ThreadSafeBase["SecureMessage"]):
    message_id: str = Field(..., description="Unique identifier for the message")
    thread_id: str = Field(..., description="Identifier for the message thread")
    patient_id: str = Field(..., description="Reference to patient")
    provider_id: str = Field(..., description="Reference to provider")
    sender_type: Literal["patient", "provider"] = Field(..., description="Type of sender")
    subject: str = Field(..., description="Subject line of the message")
    message_body: str = Field(..., description="Content of the message")
    priority: Optional[Literal["low", "normal", "high", "urgent"]] = Field(default="normal", description="Priority level of the message")
    is_read: bool = Field(default=False, description="Whether message has been read")
    delivery_status: str = Field(default="delivered", description="Status of message delivery")
    sent_at: datetime = Field(..., description="Timestamp when message was sent")
    read_at: Optional[datetime] = Field(default=None, description="Timestamp when message was marked as read")

@with_instance_key("attachment_id")
class MessageAttachment(BaseModel, ThreadSafeBase["MessageAttachment"]):
    attachment_id: str = Field(default=..., description="Unique identifier for the attachment")
    message_id: str = Field(default=..., description="Reference to message")
    document_id: str = Field(default=..., description="Reference to document")

@with_instance_key("insurance_id")
class Insurance(BaseModel, ThreadSafeBase["Insurance"]):
    insurance_id: str = Field(..., description="Unique identifier for the insurance record")
    patient_id: str = Field(..., description="Reference to patient")
    insurance_provider: str = Field(..., description="Name of the insurance company")
    policy_number: str = Field(..., description="Insurance policy number")
    group_number: Optional[str] = Field(None, description="Insurance group number")
    policy_holder_name: str = Field(..., description="Name of the policy holder")
    relationship_to_patient: Literal["self", "spouse", "parent", "child", "other"] = Field(..., description="Relationship of policy holder to patient")
    coverage_start_date: Optional[datetime] = Field(None, description="Datetime when coverage starts")
    coverage_end_date: Optional[datetime] = Field(None, description="Datetime when coverage ends")
    verification_status: str = Field("pending_verification", description="Status of insurance verification")
    added_at: datetime = Field(..., description="Timestamp when insurance was added")

@with_instance_key("payment_id")
class Payment(BaseModel, ThreadSafeBase["Payment"]):
    payment_id: str = Field(..., description="Unique identifier for the payment transaction")
    patient_id: str = Field(..., description="Reference to patient")
    appointment_id: str = Field(..., description="Reference to appointment")
    payment_amount: float = Field(..., description="Amount paid")
    payment_method: Literal["credit_card", "debit_card", "insurance", "health_savings_account"] = Field(
        ..., description="Method of payment"
    )
    card_last_four: Optional[str] = Field(None, description="Last four digits of payment card")
    transaction_status: str = Field("completed", description="Status of the payment transaction")
    receipt_number: Optional[str] = Field(None, description="Receipt number for the payment")
    processed_at: datetime = Field(..., description="Timestamp when payment was processed")

    @validator("processed_at", pre=True)
    def strip_processed_at(cls, v):
        # Remove any leading/trailing whitespace from the datetime string before parsing
        if isinstance(v, str):
            v = v.strip()
        return v

@with_instance_key("allergy_id")
class Allergy(BaseModel, ThreadSafeBase["Allergy"]):
    allergy_id: str = Field(..., description="Unique identifier for the allergy record")
    patient_id: str = Field(..., description="Reference to patient")
    allergen: str = Field(..., description="Name of the allergen")
    allergy_type: Literal["medication", "food", "environmental", "other"] = Field(
        ..., description="Type of allergy"
    )
    reaction: str = Field(..., description="Description of allergic reaction")
    severity: Literal["mild", "moderate", "severe", "life_threatening"] = Field(
        ..., description="Severity level of the allergy"
    )
    onset_date: Optional[date] = Field(
        default=None, description="Date when allergy was first identified"
    )
    added_at: datetime = Field(..., description="Timestamp when allergy was added")

    @validator("onset_date", pre=True, always=True)
    def parse_onset_date(cls, v):
        return strip_and_parse_date(v)

    @validator("added_at", pre=True, always=True)
    def parse_added_at(cls, v):
        return strip_and_parse_datetime(v)

@with_instance_key("condition_id")
class MedicalCondition(BaseModel, ThreadSafeBase["MedicalCondition"]):
    condition_id: str = Field(..., description="Unique identifier for the condition record")
    patient_id: str = Field(..., description="Reference to patient")
    condition_name: str = Field(..., description="Name of the medical condition")
    diagnosis_date: date = Field(..., description="Date when condition was diagnosed")
    condition_status: Literal["active", "resolved", "chronic", "in_remission"] = Field(
        ..., description="Current status of the condition"
    )
    severity: Optional[Literal["mild", "moderate", "severe"]] = Field(
        default=None, description="Severity of the condition"
    )
    notes: Optional[str] = Field(default=None, description="Additional notes about the condition")
    added_at: datetime = Field(..., description="Timestamp when condition was added")

    @validator("diagnosis_date", pre=True)
    def clean_diagnosis_date(cls, v):
        # Remove trailing spaces and parse with flexible datetime
        if isinstance(v, str):
            v = v.strip()
            # Try parsing as datetime first
            try:
                dt = datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
                return dt.date()
            except ValueError:
                pass
            # Try parsing as date only
            try:
                return datetime.strptime(v, "%Y-%m-%d").date()
            except ValueError:
                pass
        return v

    @validator("added_at", pre=True)
    def clean_added_at(cls, v):
        # Remove trailing spaces and parse as datetime
        if isinstance(v, str):
            v = v.strip()
            try:
                return datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                pass
            try:
                return datetime.strptime(v, "%Y-%m-%d")
            except ValueError:
                pass
        return v

@with_instance_key("vital_signs_id")
class VitalSign(BaseModel, ThreadSafeBase["VitalSign"]):
    vital_signs_id: str = Field(..., description="Unique identifier for the vital signs record")
    patient_id: str = Field(..., description="Reference to patient")
    measurement_datetime: datetime = Field(..., description="Date and time when measurements were taken")
    blood_pressure_systolic: Optional[int] = Field(default=None, description="Systolic blood pressure in mmHg")
    blood_pressure_diastolic: Optional[int] = Field(default=None, description="Diastolic blood pressure in mmHg")
    heart_rate: Optional[int] = Field(default=None, description="Heart rate in beats per minute")
    temperature: Optional[float] = Field(default=None, description="Body temperature in Fahrenheit")
    oxygen_saturation: Optional[int] = Field(default=None, description="Blood oxygen saturation percentage")
    weight: Optional[float] = Field(default=None, description="Body weight in pounds")
    recorded_at: datetime = Field(..., description="Timestamp when record was created")

@with_instance_key("rating_id")
class ProviderRating(BaseModel, ThreadSafeBase["ProviderRating"]):
    rating_id: str = Field(..., description="Unique identifier for the rating record")
    patient_id: str = Field(..., description="Reference to patient")
    provider_id: str = Field(..., description="Reference to provider")
    appointment_id: str = Field(..., description="Reference to appointment")
    overall_rating: int = Field(..., description="Overall rating from 1 to 5 stars")
    communication_rating: Optional[int] = Field(None, description="Rating for provider communication")
    professionalism_rating: Optional[int] = Field(None, description="Rating for provider professionalism")
    written_review: Optional[str] = Field(None, description="Written review comments")
    would_recommend: Optional[bool] = Field(None, description="Whether patient would recommend this provider")
    submitted_at: datetime = Field(..., description="Timestamp when rating was submitted")

@with_instance_key("preference_id")
class NotificationPreference(BaseModel, ThreadSafeBase["NotificationPreference"]):
    preference_id: str = Field(default=..., description="Unique identifier for the preference record")
    patient_id: str = Field(default=..., description="Reference to patient")
    appointment_reminders: bool = Field(default=True, description="Enable appointment reminder notifications")
    message_notifications: bool = Field(default=True, description="Enable new message notifications")
    medication_reminders: bool = Field(default=True, description="Enable medication reminder notifications")
    lab_result_notifications: bool = Field(default=True, description="Enable lab result availability notifications")
    updated_at: Optional[datetime] = Field(default=None, description="Timestamp when preferences were updated")

@with_instance_key("channel_id")
class NotificationChannel(BaseModel, ThreadSafeBase["NotificationChannel"]):
    channel_id: str = Field(..., description="Unique identifier for the channel record")
    preference_id: str = Field(..., description="Reference to notification preference")
    channel_type: Literal["email", "sms", "push_notification", "in_app"] = Field(
        ..., description="Type of notification channel"
    )

class HealthcareTelemedicineDB(DB):
    """Database containing all healthcare_telemedicine_Telehealth_Patient_Portal-related data"""
    patient: Optional[Dict[str, Patient]] = Field(
        default=None,
        description="Schema Patient"
    )
    healthcare_provider: Optional[Dict[str, HealthcareProvider]] = Field(
        default=None,
        description="Schema HealthcareProvider"
    )
    provider_specialty: Optional[Dict[str, ProviderSpecialty]] = Field(
        default=None,
        description="Schema ProviderSpecialty"
    )
    provider_language: Optional[Dict[str, ProviderLanguage]] = Field(
        default=None,
        description="Schema ProviderLanguage"
    )
    appointment: Optional[Dict[str, Appointment]] = Field(
        default=None,
        description="Schema Appointment"
    )
    medical_document: Optional[Dict[str, MedicalDocument]] = Field(
        default=None,
        description="Schema MedicalDocument"
    )
    medication: Optional[Dict[str, Medication]] = Field(
        default=None,
        description="Schema Medication"
    )
    medication_reminder: Optional[Dict[str, MedicationReminder]] = Field(
        default=None,
        description="Schema MedicationReminder"
    )
    reminder_time: Optional[Dict[str, ReminderTime]] = Field(
        default=None,
        description="Schema ReminderTime"
    )
    prescription: Optional[Dict[str, Prescription]] = Field(
        default=None,
        description="Schema Prescription"
    )
    prescription_refill_request: Optional[Dict[str, PrescriptionRefillRequest]] = Field(
        default=None,
        description="Schema PrescriptionRefillRequest"
    )
    pharmacy: Optional[Dict[str, Pharmacy]] = Field(
        default=None,
        description="Schema Pharmacy"
    )
    secure_message: Optional[Dict[str, SecureMessage]] = Field(
        default=None,
        description="Schema SecureMessage"
    )
    message_attachment: Optional[Dict[str, MessageAttachment]] = Field(
        default=None,
        description="Schema MessageAttachment"
    )
    insurance: Optional[Dict[str, Insurance]] = Field(
        default=None,
        description="Schema Insurance"
    )
    payment: Optional[Dict[str, Payment]] = Field(
        default=None,
        description="Schema Payment"
    )
    allergy: Optional[Dict[str, Allergy]] = Field(
        default=None,
        description="Schema Allergy"
    )
    medical_condition: Optional[Dict[str, MedicalCondition]] = Field(
        default=None,
        description="Schema MedicalCondition"
    )
    vital_sign: Optional[Dict[str, VitalSign]] = Field(
        default=None,
        description="Schema VitalSign"
    )
    provider_rating: Optional[Dict[str, ProviderRating]] = Field(
        default=None,
        description="Schema ProviderRating"
    )
    notification_preference: Optional[Dict[str, NotificationPreference]] = Field(
        default=None,
        description="Schema NotificationPreference"
    )
    notification_channel: Optional[Dict[str, NotificationChannel]] = Field(
        default=None,
        description="Schema NotificationChannel"
    )