"""
Database schemas for wedding_planning_Wedding_Management_Suite

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

Generated by ClassDBAgent.
"""

import json
from datetime import datetime
from datetime import datetime, date
from pathlib import Path
from pydantic import Field
from pydantic import PrivateAttr, Field
from pydantic import PrivateAttr, Field, 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("wedding_id")
class WeddingEvent(BaseModel, ThreadSafeBase["WeddingEvent"]):
    wedding_id: str = Field(default=..., description="Unique identifier for the wedding event")
    bride_name: str = Field(default=..., description="Full name of the bride")
    groom_name: str = Field(default=..., description="Full name of the groom")
    wedding_date: datetime = Field(default=..., description="Date and time of the wedding ceremony")
    venue_name: str = Field(default=..., description="Name of the wedding venue")
    wedding_theme: Optional[str] = Field(default=None, description="Theme or style of the wedding")
    estimated_guest_count: Optional[int] = Field(default=None, description="Estimated number of guests attending")
    creation_timestamp: datetime = Field(default=..., description="Timestamp when the wedding event was created")

    @validator("wedding_date", "creation_timestamp", pre=True)
    def strip_and_parse_datetime(cls, v):
        if isinstance(v, str):
            v = v.strip()
            # Accept "YYYY-MM-DD HH:MM:SS" and "YYYY-MM-DD HH:MM:SS" (without T)
            try:
                return datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                # fallback to default pydantic parsing
                return v
        return v

@with_instance_key("guest_id")
class Guest(BaseModel, ThreadSafeBase["Guest"]):
    guest_id: str = Field(default=..., description="Unique identifier for the guest")
    wedding_id: str = Field(default=..., description="Reference to the wedding event")
    guest_name: str = Field(default=..., description="Full name of the guest")
    email: str = Field(default=..., description="Email address of the guest")
    phone_number: Optional[str] = Field(default=None, description="Phone number of the guest")
    meal_preference: Optional[Literal["vegetarian", "vegan", "gluten_free", "no_restriction", "kosher", "halal"]] = Field(
        default=None, description="Dietary preference for the meal"
    )
    plus_one_allowed: Optional[bool] = Field(default=None, description="Whether the guest is allowed to bring a plus one")
    rsvp_status: Literal["attending", "declined", "pending", "maybe"] = Field(
        default="pending", description="RSVP response status"
    )
    response_date: Optional[datetime] = Field(default=None, description="Date when the RSVP was received")
    added_timestamp: datetime = Field(default=..., description="Timestamp when the guest was added")

@with_instance_key("contract_id")
class VendorContract(BaseModel, ThreadSafeBase["VendorContract"]):
    contract_id: str = Field(..., description="Unique identifier for the vendor contract")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    vendor_name: str = Field(..., description="Name of the vendor company")
    vendor_type: Literal[
        "florist", "photographer", "caterer", "dj", "videographer",
        "decorator", "baker", "planner", "transportation", "entertainment"
    ] = Field(..., description="Type of service provided by the vendor")
    contract_amount: float = Field(..., description="Total contract amount in dollars")
    deposit_amount: Optional[float] = Field(None, description="Deposit amount required in dollars")
    service_date: Optional[datetime] = Field(None, description="Date when the service will be provided")
    contact_person: Optional[str] = Field(None, description="Name of the contact person at the vendor")
    contact_phone: Optional[str] = Field(None, description="Phone number of the vendor contact")
    payment_status: Literal[
        "not_paid", "deposit_paid", "partially_paid", "fully_paid", "overdue"
    ] = Field("not_paid", description="Current payment status")
    amount_paid: float = Field(0.00, description="Amount that has been paid in dollars")
    payment_date: Optional[datetime] = Field(None, description="Date when the last payment was made")
    creation_timestamp: datetime = Field(..., description="Timestamp when the contract was created")

@with_instance_key("category_id")
class BudgetCategory(BaseModel, ThreadSafeBase["BudgetCategory"]):
    category_id: str = Field(..., description="Unique identifier for the budget category")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    category_name: Literal[
        "venue",
        "catering",
        "photography",
        "flowers",
        "music",
        "attire",
        "decorations",
        "invitations",
        "transportation",
        "honeymoon",
        "gifts",
        "miscellaneous"
    ] = Field(..., description="Name of the budget category")
    allocated_amount: float = Field(..., description="Amount allocated for this category in dollars")
    notes: Optional[str] = Field(None, description="Additional notes about this budget category")
    creation_timestamp: datetime = Field(..., description="Timestamp when the category was created")

@with_instance_key("expense_id")
class Expense(BaseModel, ThreadSafeBase["Expense"]):
    expense_id: str = Field(default=..., description="Unique identifier for the expense")
    wedding_id: str = Field(default=..., description="Reference to the wedding event")
    category_id: str = Field(default=..., description="Reference to the budget category")
    expense_description: str = Field(default=..., description="Description of the expense")
    amount: float = Field(default=..., description="Amount of the expense in dollars")
    expense_date: datetime = Field(default=..., description="Date when the expense was incurred")
    payment_method: Optional[Literal["cash", "credit_card", "debit_card", "check", "bank_transfer", "online_payment"]] = Field(
        default=None,
        description="Method used for payment"
    )

@with_instance_key("task_id")
class Task(BaseModel, ThreadSafeBase["Task"]):
    task_id: str = Field(..., description="Unique identifier for the task")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    task_title: str = Field(..., description="Title of the task")
    task_description: Optional[str] = Field(default=None, description="Detailed description of the task")
    due_date: datetime = Field(..., description="Deadline for the task")
    priority: Literal["low", "medium", "high", "urgent"] = Field(..., description="Priority level of the task")
    assigned_to: Optional[str] = Field(default=None, description="Name of the person assigned to the task")
    category: Optional[Literal["venue", "vendors", "guests", "attire", "decorations", "invitations", "logistics", "legal", "other"]] = Field(default=None, description="Category of the task")
    status: Literal["not_started", "in_progress", "completed", "cancelled", "on_hold"] = Field(default="not_started", description="Current status of the task")
    completion_date: Optional[datetime] = Field(default=None, description="Date when the task was completed")
    notes: Optional[str] = Field(default=None, description="Additional notes about the task")
    creation_timestamp: datetime = Field(..., description="Timestamp when the task was created")

@with_instance_key("table_id")
class SeatingTable(BaseModel, ThreadSafeBase["SeatingTable"]):
    table_id: str = Field(default=..., description="Unique identifier for the table")
    wedding_id: str = Field(default=..., description="Reference to the wedding event")
    table_number: str = Field(default=..., description="Number or identifier of the table")
    table_capacity: int = Field(default=..., description="Maximum number of guests that can be seated at the table")
    table_location: Optional[str] = Field(default=None, description="Location or section of the table in the venue")
    table_type: Optional[Literal["round", "rectangular", "square", "banquet", "head_table"]] = Field(
        default=None, description="Type or shape of the table"
    )
    creation_timestamp: datetime = Field(default=..., description="Timestamp when the table was created")

@with_instance_key("assignment_id")
class GuestSeatingAssignment(BaseModel, ThreadSafeBase["GuestSeatingAssignment"]):
    assignment_id: str = Field(..., description="Unique identifier for the seating assignment")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    guest_id: str = Field(..., description="Reference to the guest")
    table_id: str = Field(..., description="Reference to the table")
    seat_number: Optional[int] = Field(default=None, description="Specific seat number at the table")

@with_instance_key("timeline_event_id")
class TimelineEvent(BaseModel, ThreadSafeBase["TimelineEvent"]):
    timeline_event_id: str = Field(..., description="Unique identifier for the timeline event")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    event_name: str = Field(..., description="Name of the timeline event")
    start_time: datetime = Field(..., description="Start time of the event")
    end_time: datetime = Field(..., description="End time of the event")
    location: Optional[str] = Field(None, description="Location where the event takes place")
    description: Optional[str] = Field(None, description="Detailed description of the event")
    responsible_person: Optional[str] = Field(None, description="Person responsible for this event")
    creation_timestamp: datetime = Field(..., description="Timestamp when the event was created")

@with_instance_key("invitation_id")
class Invitation(BaseModel, ThreadSafeBase["Invitation"]):
    invitation_id: str = Field(..., description="Unique identifier for the invitation")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    guest_id: str = Field(..., description="Reference to the guest")
    delivery_method: Literal["email", "postal_mail", "hand_delivery", "digital_app"] = Field(..., description="Method used to deliver the invitation")
    invitation_message: Optional[str] = Field(None, description="Custom message included in the invitation")
    send_date: Optional[datetime] = Field(None, description="Date when the invitation is sent")
    delivery_status: str = Field(..., description="Status of the invitation delivery")
    sent_timestamp: datetime = Field(..., description="Timestamp when the invitation was sent")

@with_instance_key("registry_id")
class GiftRegistry(BaseModel, ThreadSafeBase["GiftRegistry"]):
    registry_id: str = Field(..., description="Unique identifier for the gift registry")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    registry_name: str = Field(..., description="Name of the gift registry")
    registry_url: Optional[str] = Field(None, description="URL link to the online registry")
    store_name: Optional[str] = Field(None, description="Name of the store hosting the registry")
    creation_timestamp: datetime = Field(..., description="Timestamp when the registry was created")

@with_instance_key("item_id")
class GiftItem(BaseModel, ThreadSafeBase["GiftItem"]):
    item_id: str = Field(..., description="Unique identifier for the gift item")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    registry_id: str = Field(..., description="Reference to the gift registry")
    item_name: str = Field(..., description="Name of the gift item")
    item_description: Optional[str] = Field(None, description="Description of the gift item")
    price: float = Field(..., description="Price of the item in dollars")
    quantity_desired: int = Field(..., description="Number of this item desired")
    quantity_purchased: int = Field(0, description="Number of items purchased")
    category: Optional[Literal[
        "kitchen", "bedroom", "bathroom", "dining", "home_decor",
        "electronics", "outdoor", "experience", "cash_fund"
    ]] = Field(None, description="Category of the gift item")
    purchased_by: Optional[str] = Field(None, description="Name of the guest who purchased the item")
    purchase_date: Optional[datetime] = Field(None, description="Date when the item was purchased")
    creation_timestamp: datetime = Field(..., description="Timestamp when the item was added")

    @validator("purchase_date", "creation_timestamp", pre=True)
    def strip_and_parse_datetime(cls, value):
        if isinstance(value, str):
            value = value.strip()
            # Acceptable format: 'YYYY-MM-DD HH:MM:SS'
            try:
                return datetime.strptime(value, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                # fallback to default pydantic parsing if not match
                return value
        return value

@with_instance_key("album_id")
class PhotoAlbum(BaseModel, ThreadSafeBase["PhotoAlbum"]):
    album_id: str = Field(..., description="Unique identifier for the photo album")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    album_name: str = Field(..., description="Name of the photo album")
    album_description: Optional[str] = Field(default=None, description="Description of the album")
    photographer_name: Optional[str] = Field(default=None, description="Name of the photographer")
    album_date: Optional[datetime] = Field(default=None, description="Date when photos were taken")
    creation_timestamp: datetime = Field(..., description="Timestamp when the album was created")

@with_instance_key("photo_id")
class Photo(BaseModel, ThreadSafeBase["Photo"]):
    photo_id: str = Field(..., description="Unique identifier for the photo")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    album_id: str = Field(..., description="Reference to the photo album")
    photo_url: str = Field(..., description="URL or path to the photo file")
    photo_caption: Optional[str] = Field(None, description="Caption for the photo")
    photo_timestamp: Optional[datetime] = Field(None, description="Timestamp when the photo was taken")
    tags: Optional[str] = Field(None, description="Tags or keywords for the photo stored as comma-separated values")
    upload_timestamp: datetime = Field(..., description="Timestamp when the photo was uploaded")

@with_instance_key("notification_id")
class Notification(BaseModel, ThreadSafeBase["Notification"]):
    notification_id: str = Field(default=..., description="Unique identifier for the notification")
    wedding_id: str = Field(default=..., description="Reference to the wedding event")
    recipient_type: Literal["guest", "vendor", "wedding_party", "all_guests"] = Field(
        default=..., description="Type of recipient for the reminder"
    )
    recipient_id: Optional[str] = Field(default=None, description="Unique identifier of the recipient")
    reminder_subject: str = Field(default=..., description="Subject of the reminder")
    reminder_message: str = Field(default=..., description="Content of the reminder message")
    delivery_method: Literal["email", "sms", "push_notification", "phone_call"] = Field(
        default=..., description="Method to deliver the reminder"
    )
    scheduled_send_time: Optional[datetime] = Field(default=None, description="Scheduled time to send the reminder")
    delivery_status: str = Field(default=..., description="Status of the notification delivery")

    @validator("scheduled_send_time", pre=True)
    def parse_scheduled_send_time(cls, v):
        if v is None:
            return v
        if isinstance(v, datetime):
            return v
        if isinstance(v, str):
            # Remove trailing/leading spaces and try to parse
            v = v.strip()
            try:
                return datetime.strptime(v, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                # Try ISO format fallback
                try:
                    return datetime.fromisoformat(v)
                except Exception:
                    raise ValueError("scheduled_send_time must be a valid datetime string in 'YYYY-MM-DD HH:MM:SS' format")
        return v

@with_instance_key("website_id")
class WeddingWebsite(BaseModel, ThreadSafeBase["WeddingWebsite"]):      
    website_id: str = Field(..., description="Unique identifier for the wedding website")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    website_url: str = Field(..., description="Custom URL for the wedding website")
    theme_name: Optional[Literal["classic", "modern", "rustic", "elegant", "beach", "garden", "vintage"]] = Field(default=None, description="Theme or design style for the website")
    welcome_message: Optional[str] = Field(default=None, description="Welcome message displayed on the website")
    include_rsvp_form: Optional[bool] = Field(default=None, description="Whether to include an RSVP form on the website")
    include_registry_link: Optional[bool] = Field(default=None, description="Whether to include a link to the gift registry")
    creation_timestamp: datetime = Field(..., description="Timestamp when the website was created")

@with_instance_key("playlist_id")
class MusicPlaylist(BaseModel, ThreadSafeBase["MusicPlaylist"]):      
    playlist_id: str = Field(..., description="Unique identifier for the music playlist")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    playlist_name: str = Field(..., description="Name of the music playlist")
    event_segment: Literal[
        "pre_ceremony", "processional", "ceremony", "recessional", 
        "cocktail_hour", "reception_entrance", "first_dance", "dinner", 
        "dancing", "last_song"
    ] = Field(..., description="Wedding segment for this playlist")
    total_duration_minutes: Optional[int] = Field(default=None, description="Total duration of the playlist in minutes")
    creation_timestamp: datetime = Field(..., description="Timestamp when the playlist was created")

@with_instance_key("song_id")
class Song(BaseModel, ThreadSafeBase["Song"]):
    song_id: str = Field(..., description="Unique identifier for the song")
    wedding_id: str = Field(..., description="Reference to the wedding event")
    playlist_id: str = Field(..., description="Reference to the music playlist")
    song_title: str = Field(..., description="Title of the song")
    artist_name: str = Field(..., description="Name of the artist or composer")
    duration_seconds: Optional[int] = Field(None, description="Duration of the song in seconds")
    play_order: Optional[int] = Field(None, description="Order in which the song should play in the playlist")
    added_timestamp: datetime = Field(..., description="Timestamp when the song was added")

class WeddingPlanningDB(DB):
    """Database containing all wedding_planning_Wedding_Management_Suite-related data"""
    wedding_event: Optional[Dict[str, WeddingEvent]] = Field(
        default=None,
        description="Schema WeddingEvent"
    )
    guest: Optional[Dict[str, Guest]] = Field(
        default=None,
        description="Schema Guest"
    )
    vendor_contract: Optional[Dict[str, VendorContract]] = Field(
        default=None,
        description="Schema VendorContract"
    )
    budget_category: Optional[Dict[str, BudgetCategory]] = Field(
        default=None,
        description="Schema BudgetCategory"
    )
    expense: Optional[Dict[str, Expense]] = Field(
        default=None,
        description="Schema Expense"
    )
    task: Optional[Dict[str, Task]] = Field(
        default=None,
        description="Schema Task"
    )
    seating_table: Optional[Dict[str, SeatingTable]] = Field(
        default=None,
        description="Schema SeatingTable"
    )
    guest_seating_assignment: Optional[Dict[str, GuestSeatingAssignment]] = Field(
        default=None,
        description="Schema GuestSeatingAssignment"
    )
    timeline_event: Optional[Dict[str, TimelineEvent]] = Field(
        default=None,
        description="Schema TimelineEvent"
    )
    invitation: Optional[Dict[str, Invitation]] = Field(
        default=None,
        description="Schema Invitation"
    )
    gift_registry: Optional[Dict[str, GiftRegistry]] = Field(
        default=None,
        description="Schema GiftRegistry"
    )
    gift_item: Optional[Dict[str, GiftItem]] = Field(
        default=None,
        description="Schema GiftItem"
    )
    photo_album: Optional[Dict[str, PhotoAlbum]] = Field(
        default=None,
        description="Schema PhotoAlbum"
    )
    photo: Optional[Dict[str, Photo]] = Field(
        default=None,
        description="Schema Photo"
    )
    notification: Optional[Dict[str, Notification]] = Field(
        default=None,
        description="Schema Notification"
    )
    wedding_website: Optional[Dict[str, WeddingWebsite]] = Field(
        default=None,
        description="Schema WeddingWebsite"
    )
    music_playlist: Optional[Dict[str, MusicPlaylist]] = Field(
        default=None,
        description="Schema MusicPlaylist"
    )
    song: Optional[Dict[str, Song]] = Field(
        default=None,
        description="Schema Song"
    )