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

class WeddingPlanningTools(ToolKitBase):
    """All tools for wedding_planning."""
    
    db: WeddingPlanningDB
    
    def __init__(self, db: WeddingPlanningDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def calculate_per_guest_cost(self, total_wedding_cost: float, confirmed_guest_count: int):
        """
        Calculate the average cost per guest for the wedding.

        This method computes the per-guest cost by dividing the total wedding cost
        by the number of confirmed guests. It performs input validation to ensure
        both parameters are valid and returns detailed cost breakdown information.

        Args:
            total_wedding_cost: Total cost of the wedding in dollars (must be non-negative)
            confirmed_guest_count: Number of confirmed guests attending (must be positive)

        Returns:
            dict: A dictionary containing:
                - per_guest_cost: Average cost per guest in dollars (rounded to 2 decimal places)
                - total_cost: Total wedding cost in dollars
                - guest_count: Number of guests used in calculation

        Raises:
            ZeroDivisionError: If confirmed_guest_count is zero or negative
            ValueError: If total_wedding_cost is negative
            TypeError: If inputs are not numeric types
        """

        # Validate input types
        if not isinstance(total_wedding_cost, (int, float)):
            raise TypeError(f"total_wedding_cost must be a number, got {type(total_wedding_cost).__name__}")

        if not isinstance(confirmed_guest_count, int):
            raise TypeError(f"confirmed_guest_count must be an integer, got {type(confirmed_guest_count).__name__}")

        # Validate total_wedding_cost is non-negative
        if total_wedding_cost < 0:
            raise ValueError(f"total_wedding_cost must be non-negative, got {total_wedding_cost}")

        # Validate confirmed_guest_count is positive (raises ZeroDivisionError for zero or negative)
        if confirmed_guest_count <= 0:
            raise ZeroDivisionError(f"confirmed_guest_count must be positive (greater than 0), got {confirmed_guest_count}")

        # Calculate per guest cost
        # Round to 2 decimal places for currency precision
        per_guest_cost = round(total_wedding_cost / confirmed_guest_count, 2)

        # Return the cost breakdown
        return {
            "per_guest_cost": per_guest_cost,
            "total_cost": float(total_wedding_cost),  # Ensure it's a float for consistency
            "guest_count": confirmed_guest_count
        }

    @is_tool()
    def update_wedding_date(self, wedding_id: str, new_wedding_date: str):
        """
        Update the date and time of a wedding event.

        Args:
            wedding_id: Unique identifier of the wedding event
            new_wedding_date: New date and time for the wedding in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Contains success status and updated_date

        Raises:
            ValueError: If wedding_id doesn't exist or new_wedding_date format is invalid
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Validate that wedding_event table exists
        wedding_event_table = getattr(db, "wedding_event", None)
        if wedding_event_table is None:
            raise ValueError("Wedding event table does not exist in the database")

        # Validate that the wedding_id exists in the database
        if wedding_id not in wedding_event_table:
            raise ValueError(f"Wedding event with id '{wedding_id}' does not exist")

        # Validate and parse the new_wedding_date format
        try:
            # Strip whitespace and parse the datetime string
            new_date_str = new_wedding_date.strip()
            parsed_new_date = datetime.strptime(new_date_str, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid date format for new_wedding_date. Expected 'yyyy-mm-dd HH:MM:SS', got '{new_wedding_date}'") from e

        # Validate that the new date is in the future (reasonable check for wedding planning)
        current_time = datetime.now()
        if parsed_new_date < current_time:
            raise ValueError(f"New wedding date '{new_wedding_date}' cannot be in the past")

        # Retrieve the existing wedding event
        wedding_event = wedding_event_table[wedding_id]

        # Update the wedding_date field
        wedding_event.wedding_date = parsed_new_date

        # Write the updated wedding event back to the database
        wedding_event_table[wedding_id] = wedding_event
        setattr(db, "wedding_event", wedding_event_table)

        # Format the updated date back to string format for return
        updated_date_str = parsed_new_date.strftime("%Y-%m-%d %H:%M:%S")

        # Return success response
        return {
            "success": True,
            "updated_date": updated_date_str
        }

    @is_tool()
    def create_timeline_event(self, wedding_id: str, event_name: str, start_time: str, end_time: str, location: str = None, description: str = None, responsible_person: str = None):
        # Import necessary modules
        from datetime import datetime
        import secrets
        import hashlib

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

        # Parse and validate time format (yyyy-mm-dd HH:MM:SS)
        try:
            start_datetime = datetime.strptime(start_time, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError("start_time must be in 'yyyy-mm-dd HH:MM:SS' format")

        try:
            end_datetime = datetime.strptime(end_time, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError("end_time must be in 'yyyy-mm-dd HH:MM:SS' format")

        # Validate that end_time is after start_time
        if end_datetime <= start_datetime:
            raise ValueError("end_time must be after start_time")

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

        # Generate unique timeline_event_id using hash
        prefix = "TIMELINE"
        timeline_event_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Access the database
        db = self.db

        # Get the timeline_event table, initialize if not exists
        timeline_event_table = getattr(db, "timeline_event", None)
        if timeline_event_table is None:
            timeline_event_table = {}

        # Create new TimelineEvent instance
        new_timeline_event = TimelineEvent(
            timeline_event_id=timeline_event_id,
            wedding_id=wedding_id,
            event_name=event_name,
            start_time=start_datetime,
            end_time=end_datetime,
            location=location,
            description=description,
            responsible_person=responsible_person,
            creation_timestamp=creation_datetime
        )

        # Add the new event to the table
        timeline_event_table[timeline_event_id] = new_timeline_event

        # Save the updated table back to database
        setattr(db, "timeline_event", timeline_event_table)

        # Return the result with timeline_event_id and creation_timestamp in string format
        return {
            "timeline_event_id": timeline_event_id,
            "creation_timestamp": creation_datetime.strftime("%Y-%m-%d %H:%M:%S")
        }

    @is_tool()
    def create_photo_album(self, wedding_id: str, album_name: str, album_description: str = None, photographer_name: str = None, album_date: str = None):
        """
        Create a photo album for wedding photos with metadata

        Args:
            wedding_id: Unique identifier of the wedding event
            album_name: Name of the photo album
            album_description: Description of the album (optional)
            photographer_name: Name of the photographer (optional)
            album_date: Date when photos were taken in yyyy-mm-dd HH:MM:SS format (optional)

        Returns:
            dict: Contains album_id and creation_timestamp

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

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

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

        # Parse album_date if provided
        parsed_album_date = None
        if album_date:
            if not isinstance(album_date, str):
                raise ValueError("album_date must be a string in yyyy-mm-dd HH:MM:SS format")

            try:
                # Try to parse with time first (yyyy-mm-dd HH:MM:SS)
                parsed_album_date = datetime.strptime(album_date, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                try:
                    # Try to parse date only (yyyy-mm-dd)
                    parsed_album_date = datetime.strptime(album_date, "%Y-%m-%d")
                except ValueError:
                    raise ValueError("album_date must be in yyyy-mm-dd HH:MM:SS or yyyy-mm-dd format")

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

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

        # Access the database
        db = self.db

        # Initialize photo_album table if it doesn't exist
        if db.photo_album is None:
            db.photo_album = {}

        # Create new PhotoAlbum instance
        new_album = PhotoAlbum(
            album_id=album_id,
            wedding_id=wedding_id,
            album_name=album_name,
            album_description=album_description,
            photographer_name=photographer_name,
            album_date=parsed_album_date,
            creation_timestamp=creation_timestamp
        )

        # Store the new album in the database
        db.photo_album[album_id] = new_album

        # Update the database
        setattr(db, "photo_album", db.photo_album)

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

    @is_tool()
    def mark_gift_as_purchased(
        self,
        wedding_id: str,
        item_id: str,
        purchased_by: str,
        quantity_purchased: int,
        purchase_date: str = None
    ) -> dict:
        """
        Mark a gift item as purchased by a guest.

        This method updates the purchase status of a gift item in the registry,
        including the purchaser's name, quantity purchased, and purchase date.

        Args:
            wedding_id: Unique identifier of the wedding event
            item_id: Unique identifier of the gift item
            purchased_by: Name of the guest who purchased the item
            quantity_purchased: Number of items purchased
            purchase_date: Date when the item was purchased in yyyy-mm-dd HH:MM:SS format (optional)

        Returns:
            dict: Contains success status and remaining quantity
                - success: Boolean indicating if the update was successful
                - remaining_quantity: Integer showing how many items are still needed

        Raises:
            ValueError: If gift item doesn't exist, parameters are invalid, or purchase exceeds desired quantity
        """
        from datetime import datetime

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

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

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

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

        # Parse purchase_date if provided
        purchase_datetime = None
        if purchase_date:
            if not isinstance(purchase_date, str):
                raise ValueError("purchase_date must be a string in yyyy-mm-dd HH:MM:SS format")
            try:
                # Strip whitespace and parse the date
                purchase_datetime = datetime.strptime(purchase_date.strip(), "%Y-%m-%d %H:%M:%S")
            except ValueError:
                raise ValueError("purchase_date must be in yyyy-mm-dd HH:MM:SS format (e.g., '2024-12-20 15:00:00')")
        else:
            # If no purchase_date provided, use current timestamp
            purchase_datetime = datetime.now()

        # Access the database
        db = self.db
        gift_item_table = getattr(db, "gift_item", None)

        if gift_item_table is None:
            raise ValueError("Gift item table not found in database")

        # Check if the gift item exists
        if item_id not in gift_item_table:
            raise ValueError(f"Gift item with item_id '{item_id}' does not exist in registry")

        # Retrieve the gift item
        gift_item = gift_item_table[item_id]

        # Verify the item belongs to the specified wedding
        if gift_item.wedding_id != wedding_id:
            raise ValueError(f"Gift item '{item_id}' does not belong to wedding '{wedding_id}'")

        # Calculate the new total quantity purchased
        new_quantity_purchased = gift_item.quantity_purchased + quantity_purchased

        # Verify that the purchase doesn't exceed the desired quantity
        if new_quantity_purchased > gift_item.quantity_desired:
            raise ValueError(
                f"Cannot purchase {quantity_purchased} items. "
                f"Only {gift_item.quantity_desired - gift_item.quantity_purchased} items remaining "
                f"(desired: {gift_item.quantity_desired}, already purchased: {gift_item.quantity_purchased})"
            )

        # Update the gift item with purchase information
        gift_item.quantity_purchased = new_quantity_purchased
        gift_item.purchased_by = purchased_by
        gift_item.purchase_date = purchase_datetime

        # Calculate remaining quantity needed
        remaining_quantity = gift_item.quantity_desired - new_quantity_purchased

        # Write the updated gift item back to the database
        gift_item_table[item_id] = gift_item
        setattr(db, "gift_item", gift_item_table)

        # Return success response with remaining quantity
        return {
            "success": True,
            "remaining_quantity": remaining_quantity
        }

    @is_tool()
    def calculate_budget_variance(self, wedding_id: str):
        """
        Calculate the variance between allocated budget and actual spending for each category.

        Args:
            wedding_id: Unique identifier of the wedding event

        Returns:
            Dictionary containing category variances and overall variance

        Raises:
            KeyError: If wedding_id doesn't exist or no budget categories found
        """
        # Import required modules
        from typing import Dict, List

        # Get database instance
        db = self.db

        # Retrieve budget categories for the wedding
        budget_categories = getattr(db, 'budget_category', None)
        if budget_categories is None:
            raise KeyError(f"Budget category table not found in database")

        # Filter budget categories for the specified wedding
        wedding_budget_categories = {
            cat_id: cat for cat_id, cat in budget_categories.items()
            if cat.wedding_id == wedding_id
        }

        if not wedding_budget_categories:
            raise KeyError(f"No budget categories found for wedding_id: {wedding_id}")

        # Retrieve expenses for the wedding
        expenses = getattr(db, 'expense', None)
        if expenses is None:
            # If no expense table exists, all spending is 0
            expenses = {}

        # Filter expenses for the specified wedding
        wedding_expenses = {
            exp_id: exp for exp_id, exp in expenses.items()
            if exp.wedding_id == wedding_id
        }

        # Calculate spending per category
        # Initialize spending tracker for each category
        category_spending: Dict[str, float] = {}

        for expense_id, expense in wedding_expenses.items():
            category_id = expense.category_id
            amount = expense.amount

            if category_id not in category_spending:
                category_spending[category_id] = 0.0

            category_spending[category_id] += amount

        # Build variance details for each category
        category_variances = []
        total_allocated = 0.0
        total_spent = 0.0

        for category_id, category in wedding_budget_categories.items():
            allocated = category.allocated_amount
            spent = category_spending.get(category_id, 0.0)

            # Calculate variance (positive means under budget, negative means over budget)
            variance = allocated - spent

            # Calculate variance percentage
            # Avoid division by zero
            if allocated != 0:
                variance_percentage = (variance / allocated) * 100
            else:
                # If allocated is 0 but spent is non-zero, it's infinitely over budget
                variance_percentage = -100.0 if spent > 0 else 0.0

            # Add to category variance list
            category_variances.append({
                "category_name": category.category_name,
                "allocated": allocated,
                "spent": spent,
                "variance": variance,
                "variance_percentage": round(variance_percentage, 2)
            })

            # Accumulate totals
            total_allocated += allocated
            total_spent += spent

        # Calculate overall variance
        overall_variance = total_allocated - total_spent

        # Sort category variances by category name for consistent output
        category_variances.sort(key=lambda x: x["category_name"])

        return {
            "category_variances": category_variances,
            "overall_variance": round(overall_variance, 2)
        }

    @is_tool()
    def create_task(self, wedding_id: str, task_title: str, due_date: str, priority: Literal["low", "medium", "high", "urgent"], task_description: str = None, assigned_to: str = None, category: Literal["venue", "vendors", "guests", "attire", "decorations", "invitations", "logistics", "legal", "other"] = None):
        """
        Create a wedding planning task with deadline and assignment

        This method creates a new task for a wedding event with specified details including
        title, description, deadline, priority, assignment, and category. The task is stored
        in the database with a unique identifier and creation timestamp.

        Args:
            wedding_id: Unique identifier of the wedding event
            task_title: Title of the task
            due_date: Deadline for the task in yyyy-mm-dd HH:MM:SS format
            priority: Priority level of the task (must be one of: low, medium, high, urgent)
            task_description: Detailed description of the task (optional)
            assigned_to: Name of the person assigned to the task (optional)
            category: Category of the task (optional, must be one of: venue, vendors, guests, 
                     attire, decorations, invitations, logistics, legal, other)

        Returns:
            dict: Contains task_id and creation_timestamp
                - task_id: Unique identifier for the created task
                - creation_timestamp: Timestamp when the task was created in yyyy-mm-dd HH:MM:SS format

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

        # Import the Task class for creating new task instances
        from scale_env.data_model.thread_safe_base import ThreadSafeBase

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

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

        if not due_date or not isinstance(due_date, str):
            raise ValueError("due_date is required and must be a string in yyyy-mm-dd HH:MM:SS format")

        if not priority or priority not in ["low", "medium", "high", "urgent"]:
            raise ValueError("priority is required and must be one of: low, medium, high, urgent")

        # Validate and parse due_date
        try:
            # Try full datetime format first
            if len(due_date.strip()) == 10:
                # Date only format (yyyy-mm-dd), add default time
                due_date_obj = datetime.strptime(due_date.strip(), "%Y-%m-%d")
            else:
                # Full datetime format (yyyy-mm-dd HH:MM:SS)
                due_date_obj = datetime.strptime(due_date.strip(), "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"due_date must be in yyyy-mm-dd HH:MM:SS format or yyyy-mm-dd format: {str(e)}")

        # Validate enum parameters with safety protection
        if category is not None and category not in ["venue", "vendors", "guests", "attire", "decorations", "invitations", "logistics", "legal", "other"]:
            raise ValueError(f"category must be one of: venue, vendors, guests, attire, decorations, invitations, logistics, legal, other. Got: {category}")

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

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

        # Access database
        db = self.db

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

        # Get task table
        task_table = getattr(db, "task")

        # Create new Task instance with all provided parameters
        new_task = Task(
            task_id=task_id,
            wedding_id=wedding_id,
            task_title=task_title,
            task_description=task_description,
            due_date=due_date_obj,
            priority=priority,
            assigned_to=assigned_to,
            category=category,
            status="not_started",  # Default status for new tasks
            completion_date=None,  # No completion date for new tasks
            notes=None,  # No notes initially
            creation_timestamp=creation_timestamp
        )

        # Add the new task to the database
        task_table[task_id] = new_task
        setattr(db, "task", task_table)

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

    @is_tool()
    def create_vendor_contract(
        self,
        wedding_id: str,
        vendor_name: str,
        vendor_type: Literal["florist", "photographer", "caterer", "dj", "videographer", "decorator", "baker", "planner", "transportation", "entertainment"],
        contract_amount: float,
        deposit_amount: float = None,
        service_date: str = None,
        contact_person: str = None,
        contact_phone: str = None
    ):
        """
        Create a contract record for a wedding vendor with service details and pricing.

        Args:
            wedding_id: Unique identifier of the wedding event
            vendor_name: Name of the vendor company
            vendor_type: Type of service provided by the vendor (must be one of the predefined types)
            contract_amount: Total contract amount in dollars
            deposit_amount: Optional deposit amount required in dollars
            service_date: Optional date when the service will be provided (format: yyyy-mm-dd HH:MM:SS)
            contact_person: Optional name of the contact person at the vendor
            contact_phone: Optional phone number of the vendor contact

        Returns:
            dict: Contains contract_id and creation_timestamp

        Raises:
            ValueError: If required parameters are invalid or vendor_type is not in allowed enum values
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

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

        # Validate vendor_type enum (additional safety check)
        allowed_vendor_types = ["florist", "photographer", "caterer", "dj", "videographer", 
                               "decorator", "baker", "planner", "transportation", "entertainment"]
        if vendor_type not in allowed_vendor_types:
            raise ValueError(f"vendor_type must be one of {allowed_vendor_types}, got '{vendor_type}'")

        # Validate optional deposit_amount
        if deposit_amount is not None:
            if not isinstance(deposit_amount, (int, float)) or deposit_amount < 0:
                raise ValueError("deposit_amount must be a non-negative number")
            if deposit_amount > contract_amount:
                raise ValueError("deposit_amount cannot exceed contract_amount")

        # Parse and validate service_date if provided
        service_date_obj = None
        if service_date is not None:
            if not isinstance(service_date, str):
                raise ValueError("service_date must be a string in 'yyyy-mm-dd HH:MM:SS' format")
            try:
                # Try parsing with full datetime format
                service_date_obj = datetime.strptime(service_date.strip(), "%Y-%m-%d %H:%M:%S")
            except ValueError:
                try:
                    # Try parsing with date-only format
                    service_date_obj = datetime.strptime(service_date.strip(), "%Y-%m-%d")
                except ValueError:
                    raise ValueError("service_date must be in 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd' format")

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

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

        # Access database
        db = self.db

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

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

        # Ensure contract_id is unique
        while contract_id in vendor_contract_table:
            contract_id = "VENDOR" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10].upper()

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

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

        # Create new vendor contract instance
        new_contract = VendorContract(
            contract_id=contract_id,
            wedding_id=wedding_id,
            vendor_name=vendor_name,
            vendor_type=vendor_type,
            contract_amount=float(contract_amount),
            deposit_amount=float(deposit_amount) if deposit_amount is not None else None,
            service_date=service_date_obj,
            contact_person=contact_person,
            contact_phone=contact_phone,
            payment_status="not_paid",  # Default payment status for new contracts
            amount_paid=0.00,  # No payment made yet
            payment_date=None,  # No payment date yet
            creation_timestamp=creation_timestamp
        )

        # Add contract to database
        vendor_contract_table[contract_id] = new_contract
        setattr(db, "vendor_contract", vendor_contract_table)

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

    @is_tool()
    def calculate_total_confirmed_guests(self, wedding_id: str):
        """
        Calculate the total number of guests who have confirmed attendance for a wedding event.

        This method counts guests based on their RSVP status:
        - confirmed_count: guests with rsvp_status = "attending"
        - declined_count: guests with rsvp_status = "declined"
        - pending_count: guests with rsvp_status = "pending" or "maybe"

        Args:
            wedding_id: Unique identifier of the wedding event

        Returns:
            Dictionary containing:
            - confirmed_count: Number of guests who confirmed attendance
            - declined_count: Number of guests who declined
            - pending_count: Number of guests with pending RSVP

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

        # Retrieve the guest table from the database
        guest_table = getattr(db, "guest", None)

        # Check if guest table exists
        if guest_table is None:
            raise KeyError(f"Guest table not found in database")

        # Initialize counters for different RSVP statuses
        confirmed_count = 0
        declined_count = 0
        pending_count = 0

        # Flag to check if any guest exists for this wedding
        wedding_has_guests = False

        # Iterate through all guests in the guest table
        for guest_id, guest in guest_table.items():
            # Filter guests by the specified wedding_id
            if guest.wedding_id == wedding_id:
                wedding_has_guests = True

                # Count guests based on their RSVP status
                if guest.rsvp_status == "attending":
                    confirmed_count += 1
                elif guest.rsvp_status == "declined":
                    declined_count += 1
                elif guest.rsvp_status in ["pending", "maybe"]:
                    # Both "pending" and "maybe" are considered as pending responses
                    pending_count += 1

        # If no guests found for this wedding_id, raise KeyError
        # This indicates the wedding_id does not exist or has no guests
        if not wedding_has_guests:
            raise KeyError(f"No guests found for wedding_id: {wedding_id}")

        # Return the counts as a dictionary
        return {
            "confirmed_count": confirmed_count,
            "declined_count": declined_count,
            "pending_count": pending_count
        }

    @is_tool()
    def validate_seating_capacity(self, wedding_id: str):
        """
        Validate that the seating arrangement can accommodate all confirmed guests.

        This method checks if the total seating capacity across all tables is sufficient
        to accommodate all guests who have confirmed their attendance (RSVP status = "attending").

        Args:
            wedding_id: Unique identifier of the wedding event

        Returns:
            dict: Validation result containing:
                - is_valid: Whether seating capacity is sufficient
                - total_capacity: Total seating capacity across all tables
                - confirmed_count: Number of confirmed guests
                - available_seats: Number of available seats (can be negative if over capacity)

        Raises:
            KeyError: If the wedding_id does not exist in either seating_table or guest database
        """
        # Access the database instance
        db = self.db

        # Retrieve seating tables and guests from database
        seating_tables = getattr(db, "seating_table", None)
        guests = getattr(db, "guest", None)

        # Check if wedding_id exists in seating_table
        if seating_tables is None:
            raise KeyError(f"seating_table database not found")

        # Check if any tables exist for this wedding
        wedding_tables_exist = any(
            table.wedding_id == wedding_id 
            for table in seating_tables.values()
        )

        if not wedding_tables_exist:
            raise KeyError(f"No seating tables found for wedding_id: {wedding_id}")

        # Check if wedding_id exists in guest database
        if guests is None:
            raise KeyError(f"guest database not found")

        # Check if any guests exist for this wedding
        wedding_guests_exist = any(
            guest.wedding_id == wedding_id 
            for guest in guests.values()
        )

        if not wedding_guests_exist:
            raise KeyError(f"No guests found for wedding_id: {wedding_id}")

        # Calculate total seating capacity for the wedding
        # Sum up table_capacity for all tables belonging to this wedding
        total_capacity = sum(
            table.table_capacity 
            for table in seating_tables.values() 
            if table.wedding_id == wedding_id
        )

        # Count confirmed guests (those with rsvp_status = "attending")
        # Only count guests for this specific wedding who have confirmed attendance
        confirmed_count = sum(
            1 
            for guest in guests.values() 
            if guest.wedding_id == wedding_id and guest.rsvp_status == "attending"
        )

        # Calculate available seats
        # Positive value means there are extra seats
        # Negative value means over capacity (not enough seats)
        available_seats = total_capacity - confirmed_count

        # Determine if seating arrangement is valid
        # Valid when total capacity is greater than or equal to confirmed guest count
        is_valid = total_capacity >= confirmed_count

        # Return validation result
        return {
            "is_valid": is_valid,
            "total_capacity": total_capacity,
            "confirmed_count": confirmed_count,
            "available_seats": available_seats
        }

    @is_tool()
    def create_music_playlist(
        self,
        wedding_id: str,
        playlist_name: str,
        event_segment: Literal[
            "pre_ceremony", "processional", "ceremony", "recessional",
            "cocktail_hour", "reception_entrance", "first_dance", "dinner",
            "dancing", "last_song"
        ],
        total_duration_minutes: int = None
    ):
        """
        Create a music playlist for different parts of the wedding.

        This method creates a new music playlist entry in the database for a specific
        wedding event segment. It validates the input parameters, generates a unique
        playlist ID, and stores the playlist information with a creation timestamp.

        Args:
            wedding_id: Unique identifier of the wedding event
            playlist_name: Name of the music playlist
            event_segment: Wedding segment for this playlist (must be one of the predefined segments)
            total_duration_minutes: Optional total duration of the playlist in minutes

        Returns:
            dict: Contains playlist_id and creation_timestamp
                - playlist_id: Unique identifier for the created music playlist
                - creation_timestamp: Timestamp when the playlist was created (yyyy-mm-dd HH:MM:SS format)

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

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

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

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

        # Validate event_segment against allowed enum values
        allowed_segments = [
            "pre_ceremony", "processional", "ceremony", "recessional",
            "cocktail_hour", "reception_entrance", "first_dance", "dinner",
            "dancing", "last_song"
        ]
        if event_segment not in allowed_segments:
            raise ValueError(
                f"event_segment must be one of {allowed_segments}, got '{event_segment}'"
            )

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

        # Access database
        db = self.db

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

        # Generate unique playlist ID
        prefix = "PLAYLIST"
        playlist_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10].upper()

        # Ensure the generated ID is unique
        music_playlist_table = getattr(db, "music_playlist")
        while playlist_id in music_playlist_table:
            playlist_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10].upper()

        # Get current timestamp
        creation_timestamp = datetime.now()

        # Create new MusicPlaylist instance
        new_playlist = MusicPlaylist(
            playlist_id=playlist_id,
            wedding_id=wedding_id,
            playlist_name=playlist_name,
            event_segment=event_segment,
            total_duration_minutes=total_duration_minutes,
            creation_timestamp=creation_timestamp
        )

        # Store the new playlist in the database
        music_playlist_table[playlist_id] = new_playlist
        setattr(db, "music_playlist", music_playlist_table)

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

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

        # Retrieve vendor_contract table from database
        vendor_contract_table = getattr(db, "vendor_contract", None)

        # Check if vendor_contract table exists
        if vendor_contract_table is None:
            raise KeyError(f"vendor_contract table not found in database")

        # Check if vendor_contract table is empty
        if not vendor_contract_table:
            raise KeyError(f"No vendor contracts found in database")

        # Initialize list to store matching vendors
        matching_vendors = []

        # Iterate through all vendor contracts to find matches for the given wedding_id
        for contract_id, contract in vendor_contract_table.items():
            # Check if the contract's wedding_id matches the requested wedding_id
            if contract.wedding_id == wedding_id:
                # Create vendor dictionary with required fields
                vendor_info = {
                    "contract_id": contract.contract_id,
                    "vendor_name": contract.vendor_name,
                    "vendor_type": contract.vendor_type
                }
                # Add vendor to the matching list
                matching_vendors.append(vendor_info)

        # If no vendors found for the wedding_id, raise KeyError
        if not matching_vendors:
            raise KeyError(f"No vendors found for wedding_id: {wedding_id}")

        # Calculate total number of vendors
        total_vendors = len(matching_vendors)

        # Return the vendor list and total count
        return {
            "vendors": matching_vendors,
            "total_vendors": total_vendors
        }

    @is_tool()
    def create_wedding_event(
        self,
        bride_name: str,
        groom_name: str,
        wedding_date: str,
        venue_name: str,
        wedding_theme: str = None,
        estimated_guest_count: int = None
    ):
        """
        Create a new wedding event with basic information including couple names, 
        wedding date, venue, and theme.

        This method validates the input parameters, ensures the wedding date is in the future,
        generates a unique wedding ID, and stores the wedding event in the database.
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters are not empty
        if not bride_name or not bride_name.strip():
            raise ValueError("bride_name cannot be empty")
        if not groom_name or not groom_name.strip():
            raise ValueError("groom_name cannot be empty")
        if not wedding_date or not wedding_date.strip():
            raise ValueError("wedding_date cannot be empty")
        if not venue_name or not venue_name.strip():
            raise ValueError("venue_name cannot be empty")

        # Parse and validate wedding_date format
        try:
            wedding_datetime = datetime.strptime(wedding_date.strip(), "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError("wedding_date must be in 'yyyy-mm-dd HH:MM:SS' format")

        # Check pre-condition: wedding date must be in the future
        current_time = datetime.now()
        if wedding_datetime <= current_time:
            raise ValueError("wedding_date must be in the future")

        # Validate estimated_guest_count if provided
        if estimated_guest_count is not None and estimated_guest_count < 0:
            raise ValueError("estimated_guest_count cannot be negative")

        # Generate unique wedding_id
        # Use timestamp-based ID to ensure uniqueness and readability
        date_str = wedding_datetime.strftime("%Y%m%d")
        random_suffix = secrets.token_hex(3).upper()
        wedding_id = f"WED{date_str}{random_suffix}"

        # Get current timestamp for creation_timestamp
        creation_timestamp = current_time

        # Access the database
        db = self.db

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

        # Get the wedding_event table
        wedding_event_table = getattr(db, "wedding_event")

        # Ensure wedding_id uniqueness
        while wedding_id in wedding_event_table:
            random_suffix = secrets.token_hex(3).upper()
            wedding_id = f"WED{date_str}{random_suffix}"

        # Create new WeddingEvent instance
        new_wedding = WeddingEvent(
            wedding_id=wedding_id,
            bride_name=bride_name.strip(),
            groom_name=groom_name.strip(),
            wedding_date=wedding_datetime,
            venue_name=venue_name.strip(),
            wedding_theme=wedding_theme.strip() if wedding_theme else None,
            estimated_guest_count=estimated_guest_count,
            creation_timestamp=creation_timestamp
        )

        # Store the wedding event in the database
        wedding_event_table[wedding_id] = new_wedding
        setattr(db, "wedding_event", wedding_event_table)

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

    @is_tool()
    def add_song_to_playlist(self, wedding_id: str, playlist_id: str, song_title: str, artist_name: str, duration_seconds: int = None, play_order: int = None):
        """
        Add a song to a wedding music playlist.

        This method creates a new song entry in the database associated with a specific
        wedding event and music playlist. It generates a unique song ID and records
        the timestamp when the song was added.

        Args:
            wedding_id: Unique identifier of the wedding event
            playlist_id: Unique identifier of the music playlist
            song_title: Title of the song to be added
            artist_name: Name of the artist or composer
            duration_seconds: Optional duration of the song in seconds
            play_order: Optional order in which the song should play in the playlist

        Returns:
            dict: Contains song_id and added_timestamp in "yyyy-mm-dd HH:MM:SS" format

        Raises:
            ValueError: If wedding_id, playlist_id, song_title, or artist_name is empty/invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

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

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

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

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

        # Access the database
        db = self.db

        # Get the song table from database, initialize if not exists
        song_table = getattr(db, "song", None)
        if song_table is None:
            song_table = {}
            setattr(db, "song", song_table)

        # Generate unique song ID
        prefix = "SONG"
        song_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Get current timestamp
        added_timestamp = datetime.now()

        # Create new Song object
        new_song = Song(
            song_id=song_id,
            wedding_id=wedding_id.strip(),
            playlist_id=playlist_id.strip(),
            song_title=song_title.strip(),
            artist_name=artist_name.strip(),
            duration_seconds=duration_seconds,
            play_order=play_order,
            added_timestamp=added_timestamp
        )

        # Add the song to the database
        song_table[song_id] = new_song
        setattr(db, "song", song_table)

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

        # Return the result
        return {
            "song_id": song_id,
            "added_timestamp": added_timestamp_str
        }

    @is_tool()
    def update_task_status(self, wedding_id: str, task_id: str, status: Literal["not_started", "in_progress", "completed", "cancelled", "on_hold"], completion_date: str = None, notes: str = None):
        """
        Update the completion status of a wedding planning task.

        This method updates the status of an existing task in the wedding planning system.
        It validates the task exists, updates the status and optional fields, and persists
        the changes to the database.

        Args:
            wedding_id: Unique identifier of the wedding event
            task_id: Unique identifier of the task to update
            status: New status for the task (must be one of the predefined enum values)
            completion_date: Optional date when task was completed (format: yyyy-mm-dd HH:MM:SS)
            notes: Optional additional notes about the task completion

        Returns:
            dict: Contains 'success' (bool) and 'updated_status' (str)

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

        # Get the database instance
        db = self.db

        # Retrieve the task table from database
        task_table = getattr(db, "task", None)

        # Validate that task table exists
        if task_table is None:
            raise KeyError(f"Task table not found in database")

        # Check if the task exists in the database
        if task_id not in task_table:
            raise KeyError(f"Task with task_id '{task_id}' does not exist")

        # Retrieve the existing task object
        task = task_table[task_id]

        # Verify that the task belongs to the specified wedding
        # This ensures we're updating the correct task for the right wedding
        if task.wedding_id != wedding_id:
            raise KeyError(f"Task '{task_id}' does not belong to wedding '{wedding_id}'")

        # Update the task status
        task.status = status

        # Update completion_date if provided
        # Parse the string format to datetime object for database consistency
        if completion_date is not None:
            try:
                # Parse the completion date string to datetime object
                # Support both full datetime format (yyyy-mm-dd HH:MM:SS) and date-only format (yyyy-mm-dd)
                if len(completion_date.strip()) == 10:  # Date only format
                    task.completion_date = datetime.strptime(completion_date.strip(), "%Y-%m-%d")
                else:  # Full datetime format
                    task.completion_date = datetime.strptime(completion_date.strip(), "%Y-%m-%d %H:%M:%S")
            except ValueError as e:
                raise ValueError(f"Invalid completion_date format. Expected 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd', got '{completion_date}'")

        # Update notes if provided
        if notes is not None:
            task.notes = notes

        # Persist the updated task back to the database
        # Update the task in the task_table dictionary
        task_table[task_id] = task

        # Write the updated task_table back to database
        setattr(db, "task", task_table)

        # Return success response with updated status
        return {
            "success": True,
            "updated_status": status
        }

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

        # Retrieve timeline_event table from database
        timeline_event_table = getattr(db, "timeline_event", None)

        # Check if timeline_event table exists
        if timeline_event_table is None:
            raise KeyError(f"Timeline event table not found in database")

        # Check if the table is empty
        if not timeline_event_table:
            raise KeyError(f"No timeline events found for wedding_id: {wedding_id}")

        # Filter timeline events by wedding_id
        matching_events = []
        for event_id, event in timeline_event_table.items():
            if event.wedding_id == wedding_id:
                matching_events.append(event)

        # Check if any events were found for the given wedding_id
        if not matching_events:
            raise KeyError(f"No timeline events found for wedding_id: {wedding_id}")

        # Sort events by start_time in chronological order
        matching_events.sort(key=lambda x: x.start_time)

        # Build the timeline_events list with required format
        timeline_events = []
        for event in matching_events:
            # Convert datetime objects to string format "yyyy-mm-dd HH:MM:SS"
            start_time_str = event.start_time.strftime("%Y-%m-%d %H:%M:%S")
            end_time_str = event.end_time.strftime("%Y-%m-%d %H:%M:%S")

            # Create event dictionary with all relevant information
            event_dict = {
                "event_name": event.event_name,
                "start_time": start_time_str,
                "end_time": end_time_str
            }

            # Add optional fields if they exist
            if event.location:
                event_dict["location"] = event.location
            if event.description:
                event_dict["description"] = event.description
            if event.responsible_person:
                event_dict["responsible_person"] = event.responsible_person

            timeline_events.append(event_dict)

        # Calculate total number of events
        total_events = len(timeline_events)

        # Return the complete timeline with all events and count
        return {
            "timeline_events": timeline_events,
            "total_events": total_events
        }

    @is_tool()
    def add_photo_to_album(
        self,
        wedding_id: str,
        album_id: str,
        photo_url: str,
        photo_caption: str = None,
        photo_timestamp: str = None,
        tags: List[str] = None
    ):
        """
        Add a photo to a wedding album with metadata and tags.

        This method creates a new photo entry in the database with the provided metadata,
        generates a unique photo ID, and stores the photo information including URL, caption,
        timestamp, and tags.
        """
        # Import required modules
        import secrets
        import hashlib
        from datetime import datetime

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

        # Access the database
        db = self.db

        # Get the photo table from database, initialize if not exists
        photo_table = getattr(db, "photo", None)
        if photo_table is None:
            photo_table = {}

        # Generate unique photo ID with prefix "PHOTO"
        photo_id = "PHOTO" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Ensure the generated photo_id is unique
        while photo_id in photo_table:
            photo_id = "PHOTO" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp for upload_timestamp in datetime format
        upload_timestamp = datetime.now()

        # Parse photo_timestamp if provided, otherwise set to None
        parsed_photo_timestamp = None
        if photo_timestamp:
            try:
                # Parse the timestamp string in "yyyy-mm-dd HH:MM:SS" format
                parsed_photo_timestamp = datetime.strptime(photo_timestamp, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                # If full datetime format fails, try date-only format
                try:
                    parsed_photo_timestamp = datetime.strptime(photo_timestamp, "%Y-%m-%d")
                except ValueError:
                    raise ValueError(f"Invalid photo_timestamp format: {photo_timestamp}. Expected 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd'")

        # Convert tags list to comma-separated string for storage
        tags_str = None
        if tags:
            # Filter out empty strings and join with comma
            valid_tags = [tag.strip() for tag in tags if tag and tag.strip()]
            if valid_tags:
                tags_str = ",".join(valid_tags)

        # Create new Photo object with all provided information
        new_photo = Photo(
            photo_id=photo_id,
            wedding_id=wedding_id,
            album_id=album_id,
            photo_url=photo_url,
            photo_caption=photo_caption,
            photo_timestamp=parsed_photo_timestamp,
            tags=tags_str,
            upload_timestamp=upload_timestamp
        )

        # Add the new photo to the photo table
        photo_table[photo_id] = new_photo

        # Update the database with the modified photo table
        setattr(db, "photo", photo_table)

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

        # Return the photo_id and upload_timestamp
        return {
            "photo_id": photo_id,
            "upload_timestamp": upload_timestamp_str
        }

    @is_tool()
    def add_guest_to_wedding(
        self,
        wedding_id: str,
        guest_name: str,
        email: str,
        phone_number: str = None,
        meal_preference: Literal["vegetarian", "vegan", "gluten_free", "no_restriction", "kosher", "halal"] = None,
        plus_one_allowed: bool = None
    ):
        """
        Add a new guest to the wedding guest list with contact information and meal preferences.

        This method creates a new guest record in the database with the provided information.
        It validates the wedding_id exists, generates a unique guest_id, and stores the guest
        with the current timestamp.
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

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

        # Validate email format (basic validation)
        if "@" not in email or "." not in email.split("@")[-1]:
            raise ValueError(f"Invalid email format: {email}")

        # Validate meal_preference enum if provided
        valid_meal_preferences = ["vegetarian", "vegan", "gluten_free", "no_restriction", "kosher", "halal"]
        if meal_preference is not None and meal_preference not in valid_meal_preferences:
            raise ValueError(
                f"Invalid meal_preference: {meal_preference}. "
                f"Must be one of {valid_meal_preferences}"
            )

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

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

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

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

        # Create new Guest instance
        new_guest = Guest(
            guest_id=guest_id,
            wedding_id=wedding_id,
            guest_name=guest_name,
            email=email,
            phone_number=phone_number,
            meal_preference=meal_preference,
            plus_one_allowed=plus_one_allowed,
            rsvp_status="pending",  # Default RSVP status
            response_date=None,  # No response yet
            added_timestamp=added_timestamp
        )

        # Add the new guest to the guest table
        guest_table[guest_id] = new_guest
        setattr(db, "guest", guest_table)

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

        # Return the guest_id and added_timestamp
        return {
            "guest_id": guest_id,
            "added_timestamp": added_timestamp_str
        }

    @is_tool()
    def update_vendor_payment_status(
        self,
        wedding_id: str,
        contract_id: str,
        payment_status: Literal["not_paid", "deposit_paid", "partially_paid", "fully_paid", "overdue"],
        amount_paid: float,
        payment_date: str = None
    ) -> dict:
        """
        Update the payment status for a vendor contract.

        This method updates the payment information for a specific vendor contract,
        including the payment status, amount paid, and payment date. It calculates
        the remaining balance and validates that the contract exists.

        Args:
            wedding_id: Unique identifier of the wedding event
            contract_id: Unique identifier of the vendor contract
            payment_status: Current payment status (must be one of the enum values)
            amount_paid: Amount that has been paid in dollars
            payment_date: Date when the payment was made in yyyy-mm-dd HH:MM:SS format (optional)

        Returns:
            dict: Contains success status and remaining balance
                - success: Boolean indicating whether the update was successful
                - remaining_balance: Remaining balance to be paid in dollars

        Raises:
            ValueError: If contract doesn't exist, wedding_id doesn't match, 
                       amount_paid is negative, or payment_date format is invalid
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Validate amount_paid is non-negative
        if amount_paid < 0:
            raise ValueError(f"Amount paid cannot be negative: {amount_paid}")

        # Retrieve the vendor_contract table from database
        vendor_contract_table = getattr(db, "vendor_contract", None)

        # Check if vendor_contract table exists
        if vendor_contract_table is None:
            raise ValueError("Vendor contract table does not exist in database")

        # Check if the contract exists
        if contract_id not in vendor_contract_table:
            raise ValueError(f"Vendor contract with contract_id '{contract_id}' does not exist")

        # Retrieve the specific contract
        contract = vendor_contract_table[contract_id]

        # Verify the wedding_id matches
        if contract.wedding_id != wedding_id:
            raise ValueError(
                f"Wedding ID mismatch: contract '{contract_id}' belongs to wedding '{contract.wedding_id}', "
                f"not '{wedding_id}'"
            )

        # Parse and validate payment_date if provided
        payment_datetime = None
        if payment_date is not None:
            try:
                # Parse the payment date string to datetime object
                payment_datetime = datetime.strptime(payment_date, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                # Try parsing date-only format if full datetime format fails
                try:
                    payment_datetime = datetime.strptime(payment_date, "%Y-%m-%d")
                except ValueError:
                    raise ValueError(
                        f"Invalid payment_date format: '{payment_date}'. "
                        f"Expected format: 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd'"
                    )

        # Update the contract fields
        contract.payment_status = payment_status
        contract.amount_paid = amount_paid

        # Update payment_date if provided
        if payment_datetime is not None:
            contract.payment_date = payment_datetime

        # Calculate remaining balance
        # Remaining balance = total contract amount - amount paid
        remaining_balance = contract.contract_amount - amount_paid

        # Ensure remaining balance is not negative (in case of overpayment, set to 0)
        if remaining_balance < 0:
            remaining_balance = 0.0

        # Update the contract in the database
        vendor_contract_table[contract_id] = contract
        setattr(db, "vendor_contract", vendor_contract_table)

        # Return success status and remaining balance
        return {
            "success": True,
            "remaining_balance": round(remaining_balance, 2)  # Round to 2 decimal places for currency
        }

    @is_tool()
    def calculate_days_until_wedding(self, wedding_date: str, current_date: str):
        """
        Calculate the number of days, weeks, and months remaining until the wedding date.

        This method computes the time difference between the current date and the wedding date,
        returning the remaining time in days, weeks, and months.

        Args:
            wedding_date: Date of the wedding in yyyy-mm-dd HH:MM:SS format
            current_date: Current date for calculation in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Dictionary containing:
                - days_remaining: Number of days until the wedding
                - weeks_remaining: Number of weeks until the wedding
                - months_remaining: Number of months until the wedding

        Raises:
            ValueError: If date strings are invalid or wedding_date is before current_date
        """
        from datetime import datetime

        # Validate and parse wedding_date
        try:
            wedding_dt = datetime.strptime(wedding_date.strip(), "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid wedding_date format. Expected 'yyyy-mm-dd HH:MM:SS', got '{wedding_date}': {str(e)}")

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

        # Ensure wedding_date is in the future relative to current_date
        if wedding_dt < current_dt:
            raise ValueError(f"Wedding date ({wedding_date}) must be after current date ({current_date})")

        # Calculate the time difference
        time_delta = wedding_dt - current_dt

        # Calculate days remaining (total days in the timedelta)
        days_remaining = time_delta.days

        # Calculate weeks remaining (floor division by 7)
        weeks_remaining = days_remaining // 7

        # Calculate months remaining (approximate calculation)
        # We calculate the difference in years and months between the two dates
        years_diff = wedding_dt.year - current_dt.year
        months_diff = wedding_dt.month - current_dt.month

        # Total months difference
        total_months = years_diff * 12 + months_diff

        # Adjust if the day of the month in wedding_dt is less than current_dt
        # This means we haven't completed a full month yet
        if wedding_dt.day < current_dt.day:
            total_months -= 1

        # Ensure months_remaining is non-negative
        months_remaining = max(0, total_months)

        # Return the calculated results
        return {
            "days_remaining": days_remaining,
            "weeks_remaining": weeks_remaining,
            "months_remaining": months_remaining
        }

    @is_tool()
    def get_tasks_by_deadline(self, wedding_id: str, start_date: str, end_date: str):
        """
        Retrieve all tasks due within a specified date range for a given wedding event.

        Args:
            wedding_id: Unique identifier of the wedding event
            start_date: Start date of the range in yyyy-mm-dd HH:MM:SS format
            end_date: End date of the range in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Contains 'tasks' (list of task dictionaries) and 'task_count' (integer)

        Raises:
            ValueError: If parameters are invalid or wedding doesn't exist
        """
        from datetime import datetime

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

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

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

        # Parse date strings to datetime objects
        try:
            start_datetime = datetime.strptime(start_date, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"start_date must be in format 'yyyy-mm-dd HH:MM:SS', got: {start_date}")

        try:
            end_datetime = datetime.strptime(end_date, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"end_date must be in format 'yyyy-mm-dd HH:MM:SS', got: {end_date}")

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

        # Access the database
        db = self.db

        # Get task table from database
        task_table = getattr(db, "task", None)

        # Check if task table exists
        if task_table is None:
            raise ValueError("Task table does not exist in database")

        # Check if any tasks exist for the given wedding_id (pre-condition: wedding event must exist)
        wedding_tasks_exist = any(
            task.wedding_id == wedding_id 
            for task in task_table.values()
        )

        if not wedding_tasks_exist and len(task_table) > 0:
            # If there are tasks in the system but none for this wedding_id, the wedding might not exist
            raise ValueError(f"No tasks found for wedding_id: {wedding_id}. Wedding event may not exist.")

        # Filter tasks by wedding_id and deadline range
        matching_tasks = []

        for task_id, task in task_table.items():
            # Check if task belongs to the specified wedding
            if task.wedding_id != wedding_id:
                continue

            # Check if task's due_date falls within the specified range
            # Note: due_date in database is already a datetime object
            if start_datetime <= task.due_date <= end_datetime:
                # Convert task to dictionary format for return
                task_dict = {
                    "task_id": task.task_id,
                    "task_title": task.task_title,
                    "due_date": task.due_date.strftime("%Y-%m-%d %H:%M:%S")
                }

                # Include optional fields if they exist
                if task.task_description is not None:
                    task_dict["task_description"] = task.task_description

                task_dict["priority"] = task.priority

                if task.assigned_to is not None:
                    task_dict["assigned_to"] = task.assigned_to

                if task.category is not None:
                    task_dict["category"] = task.category

                task_dict["status"] = task.status

                if task.completion_date is not None:
                    task_dict["completion_date"] = task.completion_date.strftime("%Y-%m-%d %H:%M:%S")

                if task.notes is not None:
                    task_dict["notes"] = task.notes

                task_dict["creation_timestamp"] = task.creation_timestamp.strftime("%Y-%m-%d %H:%M:%S")
                task_dict["wedding_id"] = task.wedding_id

                matching_tasks.append(task_dict)

        # Sort tasks by due_date (earliest first) for better usability
        matching_tasks.sort(key=lambda x: x["due_date"])

        # Return results with task list and count
        return {
            "tasks": matching_tasks,
            "task_count": len(matching_tasks)
        }

    @is_tool()
    def calculate_guest_meal_summary(self, wedding_id: str):
        """
        Calculate the summary of meal preferences across all guests for a specific wedding event.

        Args:
            wedding_id: Unique identifier of the wedding event

        Returns:
            dict: Contains meal_counts (dict of meal type to count) and total_guests_with_preferences (int)

        Raises:
            KeyError: If the wedding_id does not exist or no guests are found for the wedding
        """
        # Access the database
        db = self.db

        # Get the guest table from database
        guest_table = getattr(db, "guest", None)

        # Validate that guest table exists
        if guest_table is None:
            raise KeyError(f"Guest table not found in database")

        # Filter guests by wedding_id
        wedding_guests = []
        for guest_id, guest in guest_table.items():
            if guest.wedding_id == wedding_id:
                wedding_guests.append(guest)

        # Check if any guests exist for this wedding
        if not wedding_guests:
            raise KeyError(f"No guests found for wedding_id: {wedding_id}")

        # Initialize meal counts dictionary with all possible meal preferences
        meal_counts = {
            "vegetarian": 0,
            "vegan": 0,
            "gluten_free": 0,
            "no_restriction": 0,
            "kosher": 0,
            "halal": 0
        }

        # Count guests with meal preferences (non-None values)
        total_guests_with_preferences = 0

        # Iterate through wedding guests and count meal preferences
        for guest in wedding_guests:
            # Check if guest has a meal preference recorded
            if guest.meal_preference is not None:
                total_guests_with_preferences += 1

                # Increment the count for this meal preference
                if guest.meal_preference in meal_counts:
                    meal_counts[guest.meal_preference] += 1

        # Remove meal types with zero count for cleaner output
        # (keeping only meal types that have at least one guest)
        meal_counts = {meal_type: count for meal_type, count in meal_counts.items() if count > 0}

        # Return the summary
        return {
            "meal_counts": meal_counts,
            "total_guests_with_preferences": total_guests_with_preferences
        }

    @is_tool()
    def get_guest_list(self, wedding_id: str):
        """
        Retrieve the complete list of guests for a wedding event.

        This method fetches all guests associated with a specific wedding ID from the database
        and returns them in a structured format with the total count.

        Args:
            wedding_id: Unique identifier of the wedding event

        Returns:
            dict: Contains 'guests' (list of guest dictionaries) and 'total_count' (integer)

        Raises:
            KeyError: If the wedding_id does not exist or no guests table is found
        """
        # Access the database instance
        db = self.db

        # Retrieve the guest table from the database
        guest_table = getattr(db, 'guest', None)

        # Validate that the guest table exists
        if guest_table is None:
            raise KeyError(f"Guest table not found in database")

        # Initialize list to store matching guests
        matching_guests = []

        # Iterate through all guests in the table
        for guest_id, guest_obj in guest_table.items():
            # Check if the guest belongs to the specified wedding
            if guest_obj.wedding_id == wedding_id:
                # Convert guest object to dictionary format
                guest_dict = {
                    'guest_id': guest_obj.guest_id,
                    'guest_name': guest_obj.guest_name,
                    'email': guest_obj.email,
                    'phone_number': guest_obj.phone_number,
                    'meal_preference': guest_obj.meal_preference,
                    'plus_one_allowed': guest_obj.plus_one_allowed,
                    'rsvp_status': guest_obj.rsvp_status,
                    'response_date': guest_obj.response_date.strftime('%Y-%m-%d %H:%M:%S') if guest_obj.response_date else None,
                    'added_timestamp': guest_obj.added_timestamp.strftime('%Y-%m-%d %H:%M:%S')
                }
                matching_guests.append(guest_dict)

        # Check if any guests were found for this wedding
        if len(matching_guests) == 0:
            raise KeyError(f"No guests found for wedding_id: {wedding_id}")

        # Calculate total count of guests
        total_count = len(matching_guests)

        # Return the result in the specified format
        return {
            'guests': matching_guests,
            'total_count': total_count
        }

    @is_tool()
    def calculate_total_vendor_cost(self, wedding_id: str):
        """
        Calculate the total cost of all vendors for a wedding.

        This method retrieves all vendor contracts associated with a specific wedding
        and calculates:
        - Total contract amount across all vendors
        - Total amount already paid
        - Total remaining balance

        Args:
            wedding_id: Unique identifier of the wedding event

        Returns:
            dict: Contains total_contract_amount, total_paid, and total_remaining

        Raises:
            KeyError: If the wedding_id does not exist in any vendor contracts
        """
        # Access the database
        db = self.db

        # Get the vendor_contract table
        vendor_contract_table = getattr(db, "vendor_contract", None)

        # Check if vendor_contract table exists and is not empty
        if vendor_contract_table is None or not vendor_contract_table:
            raise KeyError(f"No vendor contracts found in the database")

        # Initialize totals
        total_contract_amount = 0.0
        total_paid = 0.0
        found_wedding = False

        # Iterate through all vendor contracts to find matching wedding_id
        for contract_id, contract in vendor_contract_table.items():
            # Check if this contract belongs to the specified wedding
            if contract.wedding_id == wedding_id:
                found_wedding = True

                # Add contract amount to total
                total_contract_amount += contract.contract_amount

                # Add paid amount to total
                # amount_paid defaults to 0.00 if not set
                total_paid += contract.amount_paid

        # If no contracts found for this wedding_id, raise KeyError
        if not found_wedding:
            raise KeyError(f"No vendor contracts found for wedding_id: {wedding_id}")

        # Calculate total remaining balance
        # This is the difference between total contract amount and total paid
        total_remaining = total_contract_amount - total_paid

        # Return the calculated totals
        return {
            "total_contract_amount": total_contract_amount,
            "total_paid": total_paid,
            "total_remaining": total_remaining
        }

    @is_tool()
    def send_invitation(self, wedding_id: str, guest_id: str, delivery_method: Literal["email", "postal_mail", "hand_delivery", "digital_app"], invitation_message: str = None, send_date: str = None):
        """
        Send a wedding invitation to a guest via specified delivery method.

        Args:
            wedding_id: Unique identifier of the wedding event
            guest_id: Unique identifier of the guest
            delivery_method: Method used to deliver the invitation (must be one of: email, postal_mail, hand_delivery, digital_app)
            invitation_message: Custom message to include in the invitation (optional)
            send_date: Date when the invitation is sent in yyyy-mm-dd HH:MM:SS format (optional, defaults to current time)

        Returns:
            dict: Contains invitation_id, delivery_status, and sent_timestamp

        Raises:
            ValueError: If delivery_method is not valid, or if required parameters are missing
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate required parameters
        if not wedding_id:
            raise ValueError("wedding_id is required")
        if not guest_id:
            raise ValueError("guest_id is required")
        if not delivery_method:
            raise ValueError("delivery_method is required")

        # Validate delivery_method against allowed enum values
        valid_delivery_methods = ["email", "postal_mail", "hand_delivery", "digital_app"]
        if delivery_method not in valid_delivery_methods:
            raise ValueError(f"Invalid delivery_method '{delivery_method}'. Must be one of: {', '.join(valid_delivery_methods)}")

        # Access the database
        db = self.db

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

        # Generate unique invitation ID
        prefix = "INV"
        invitation_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Parse send_date or use current timestamp
        if send_date:
            try:
                # Parse the provided send_date string
                send_datetime = datetime.strptime(send_date, "%Y-%m-%d %H:%M:%S")
            except ValueError:
                raise ValueError(f"Invalid send_date format '{send_date}'. Expected format: yyyy-mm-dd HH:MM:SS")
        else:
            # Use current timestamp if send_date is not provided
            send_datetime = datetime.now()

        # Create the invitation object
        new_invitation = Invitation(
            invitation_id=invitation_id,
            wedding_id=wedding_id,
            guest_id=guest_id,
            delivery_method=delivery_method,
            invitation_message=invitation_message,
            send_date=send_datetime,
            delivery_status="sent",  # Set initial status as "sent"
            sent_timestamp=send_datetime
        )

        # Add the new invitation to the database
        invitation_table[invitation_id] = new_invitation
        setattr(db, "invitation", invitation_table)

        # Format the sent_timestamp for return (convert datetime to string)
        sent_timestamp_str = send_datetime.strftime("%Y-%m-%d %H:%M:%S")

        # Return the result
        return {
            "invitation_id": invitation_id,
            "delivery_status": "sent",
            "sent_timestamp": sent_timestamp_str
        }

    @is_tool()
    def calculate_task_completion_rate(self, wedding_id: str):
        """
        Calculate the percentage of completed tasks for the wedding.

        This method analyzes all tasks associated with a specific wedding event
        and computes statistics including total tasks, completed tasks, 
        completion rate, in-progress tasks, and not-started tasks.

        Args:
            wedding_id: Unique identifier of the wedding event

        Returns:
            Dictionary containing task completion statistics

        Raises:
            KeyError: If the wedding_id has no associated tasks
        """
        # Access the database
        db = self.db

        # Get the task table from database
        task_table = getattr(db, "task", None)

        # Check if task table exists
        if task_table is None:
            raise KeyError(f"Task table not found in database")

        # Filter tasks by wedding_id
        wedding_tasks = []
        for task_id, task in task_table.items():
            if task.wedding_id == wedding_id:
                wedding_tasks.append(task)

        # Check if any tasks exist for this wedding
        if not wedding_tasks:
            raise KeyError(f"No tasks found for wedding_id: {wedding_id}")

        # Initialize counters
        total_tasks = len(wedding_tasks)
        completed_tasks = 0
        in_progress_tasks = 0
        not_started_tasks = 0

        # Count tasks by status
        for task in wedding_tasks:
            if task.status == "completed":
                completed_tasks += 1
            elif task.status == "in_progress":
                in_progress_tasks += 1
            elif task.status == "not_started":
                not_started_tasks += 1
            # Note: "cancelled" and "on_hold" statuses are not counted in the return schema
            # but are included in total_tasks count

        # Calculate completion rate
        # Avoid division by zero (though we already checked total_tasks > 0)
        if total_tasks > 0:
            completion_rate = round((completed_tasks / total_tasks) * 100, 2)
        else:
            completion_rate = 0.0

        # Return the statistics
        return {
            "total_tasks": total_tasks,
            "completed_tasks": completed_tasks,
            "completion_rate": completion_rate,
            "in_progress_tasks": in_progress_tasks,
            "not_started_tasks": not_started_tasks
        }

    @is_tool()
    def send_reminder_notification(
        self,
        wedding_id: str,
        recipient_type: Literal["guest", "vendor", "wedding_party", "all_guests"],
        reminder_subject: str,
        reminder_message: str,
        delivery_method: Literal["email", "sms", "push_notification", "phone_call"],
        recipient_id: str = None,
        scheduled_send_time: str = None
    ):
        """
        Send reminder notifications to guests or vendors about upcoming events or deadlines.

        This method creates a notification record in the database with the provided details.
        If scheduled_send_time is provided, the notification will be scheduled for that time,
        otherwise it will be sent immediately.
        """
        # Import required modules
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

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

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

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

        # Validate recipient_type enum (with safety protection)
        valid_recipient_types = ["guest", "vendor", "wedding_party", "all_guests"]
        if recipient_type not in valid_recipient_types:
            raise ValueError(f"recipient_type must be one of {valid_recipient_types}, got '{recipient_type}'")

        # Validate delivery_method enum (with safety protection)
        valid_delivery_methods = ["email", "sms", "push_notification", "phone_call"]
        if delivery_method not in valid_delivery_methods:
            raise ValueError(f"delivery_method must be one of {valid_delivery_methods}, got '{delivery_method}'")

        # Validate recipient_id if provided (for specific recipient types)
        # Note: recipient_id is optional when recipient_type is "all_guests"
        if recipient_type != "all_guests" and not recipient_id:
            raise ValueError(f"recipient_id is required when recipient_type is '{recipient_type}'")

        # Parse and validate scheduled_send_time if provided
        scheduled_datetime = None
        if scheduled_send_time:
            if not isinstance(scheduled_send_time, str):
                raise ValueError("scheduled_send_time must be a string in 'yyyy-mm-dd HH:MM:SS' format")

            try:
                # Parse the datetime string
                scheduled_datetime = datetime.strptime(scheduled_send_time.strip(), "%Y-%m-%d %H:%M:%S")
            except ValueError:
                raise ValueError("scheduled_send_time must be in 'yyyy-mm-dd HH:MM:SS' format (e.g., '2025-01-10 09:00:00')")

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

        # Determine delivery status based on scheduled time
        # If scheduled_send_time is provided and in the future, status is "scheduled"
        # Otherwise, status is "sent" (immediate delivery)
        current_time = datetime.now()
        if scheduled_datetime and scheduled_datetime > current_time:
            delivery_status = "scheduled"
        else:
            delivery_status = "sent"
            # If scheduled time is in the past or not provided, treat as immediate send
            if not scheduled_datetime:
                scheduled_datetime = current_time

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

        # Create new notification record
        from datetime import datetime
        new_notification = Notification(
            notification_id=notification_id,
            wedding_id=wedding_id,
            recipient_type=recipient_type,
            recipient_id=recipient_id,
            reminder_subject=reminder_subject,
            reminder_message=reminder_message,
            delivery_method=delivery_method,
            scheduled_send_time=scheduled_datetime,
            delivery_status=delivery_status
        )

        # Add notification to the table
        notification_table[notification_id] = new_notification

        # Save the updated table back to database
        setattr(db, "notification", notification_table)

        # Return the notification details
        return {
            "notification_id": notification_id,
            "delivery_status": delivery_status
        }

    @is_tool()
    def get_wedding_details(self, wedding_id: str):
        """
        Retrieve complete details of a wedding event by its identifier.

        This method fetches a wedding event from the database using the provided wedding_id
        and returns all relevant details including bride/groom names, date, venue, theme,
        and guest count.

        Args:
            wedding_id: Unique identifier of the wedding event to retrieve

        Returns:
            dict: A dictionary containing complete wedding details with the following keys:
                - wedding_id: Unique identifier of the wedding event
                - bride_name: Full name of the bride
                - groom_name: Full name of the groom
                - wedding_date: Date and time of the wedding in "yyyy-mm-dd HH:MM:SS" format
                - venue_name: Name of the wedding venue
                - wedding_theme: Theme or style of the wedding (may be None)
                - estimated_guest_count: Estimated number of guests (may be None)

        Raises:
            KeyError: If the wedding event with the specified wedding_id does not exist
        """
        from datetime import datetime

        # Access the database instance
        db = self.db

        # Retrieve the wedding_event table from the database
        wedding_event_table = getattr(db, "wedding_event", None)

        # Check if the wedding_event table exists
        if wedding_event_table is None:
            raise KeyError(f"Wedding event table does not exist in the database")

        # Check if the wedding_id exists in the table
        if wedding_id not in wedding_event_table:
            raise KeyError(f"Wedding event with ID '{wedding_id}' does not exist")

        # Retrieve the wedding event object
        wedding = wedding_event_table[wedding_id]

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

        # Construct and return the wedding details dictionary
        wedding_details = {
            "wedding_id": wedding.wedding_id,
            "bride_name": wedding.bride_name,
            "groom_name": wedding.groom_name,
            "wedding_date": wedding_date_str,
            "venue_name": wedding.venue_name,
            "wedding_theme": wedding.wedding_theme,
            "estimated_guest_count": wedding.estimated_guest_count
        }

        return wedding_details

    @is_tool()
    def generate_rehearsal_schedule(
        self,
        wedding_date: str,
        rehearsal_date: str,
        venue_name: str = None,
        wedding_party_size: int = None
    ) -> dict:
        """
        Generate a comprehensive schedule for the wedding rehearsal with all participants and activities.

        This method creates a structured timeline of rehearsal activities based on the wedding date,
        rehearsal date, venue information, and wedding party size. The schedule includes standard
        rehearsal activities with appropriate timing and duration.

        Args:
            wedding_date: Date of the wedding in "yyyy-mm-dd HH:MM:SS" format
            rehearsal_date: Date of the rehearsal in "yyyy-mm-dd HH:MM:SS" format
            venue_name: Optional name of the venue for the rehearsal
            wedding_party_size: Optional number of people in the wedding party

        Returns:
            dict: Contains:
                - rehearsal_activities: List of activity dictionaries with activity name, start_time, and duration_minutes
                - total_duration_minutes: Total duration of the rehearsal in minutes

        Raises:
            ValueError: If date formats are invalid, dates are in the past, or rehearsal is not before wedding
        """
        from datetime import datetime, timedelta

        # Validate and parse input dates
        try:
            wedding_dt = datetime.strptime(wedding_date.strip(), "%Y-%m-%d %H:%M:%S")
        except (ValueError, AttributeError) as e:
            raise ValueError(f"Invalid wedding_date format. Expected 'yyyy-mm-dd HH:MM:SS', got: {wedding_date}")

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

        # Validate that rehearsal is before the wedding
        if rehearsal_dt >= wedding_dt:
            raise ValueError("Rehearsal date must be before the wedding date")

        # Validate that dates are not in the past (optional business logic)
        current_time = datetime.now()
        if rehearsal_dt < current_time:
            raise ValueError("Rehearsal date cannot be in the past")

        # Determine party size for scheduling (default to 10 if not provided)
        party_size = wedding_party_size if wedding_party_size is not None else 10

        # Validate party size
        if party_size < 2:
            raise ValueError("Wedding party size must be at least 2")

        # Calculate dynamic durations based on party size
        # Larger parties need more time for processional/recessional practice
        base_processional_duration = 30
        processional_duration = base_processional_duration + (party_size // 4) * 5  # Add 5 min per 4 people

        base_recessional_duration = 15
        recessional_duration = base_recessional_duration + (party_size // 4) * 3  # Add 3 min per 4 people

        # Define standard rehearsal activities with calculated durations
        # Activities are ordered in typical rehearsal sequence
        activities_template = [
            {
                "activity": "Welcome and introduction",
                "duration_minutes": 10,
                "description": "Gather all participants and explain rehearsal objectives"
            },
            {
                "activity": "Venue walkthrough",
                "duration_minutes": 15,
                "description": "Tour the ceremony space and identify key positions"
            },
            {
                "activity": "Processional practice",
                "duration_minutes": processional_duration,
                "description": "Practice the entrance order, spacing, and timing"
            },
            {
                "activity": "Ceremony positioning",
                "duration_minutes": 20,
                "description": "Practice standing positions and movements during ceremony"
            },
            {
                "activity": "Vow exchange practice",
                "duration_minutes": 15,
                "description": "Rehearse the vow exchange and ring ceremony"
            },
            {
                "activity": "Recessional practice",
                "duration_minutes": recessional_duration,
                "description": "Practice the exit order and timing"
            },
            {
                "activity": "Special moments rehearsal",
                "duration_minutes": 20,
                "description": "Practice any special ceremonies (unity candle, sand ceremony, etc.)"
            },
            {
                "activity": "Photography positioning",
                "duration_minutes": 15,
                "description": "Review photo opportunities and groupings"
            },
            {
                "activity": "Q&A and final walkthrough",
                "duration_minutes": 15,
                "description": "Address questions and do final run-through"
            }
        ]

        # Build the schedule with actual start times
        rehearsal_activities = []
        current_time = rehearsal_dt

        for activity_template in activities_template:
            activity_entry = {
                "activity": activity_template["activity"],
                "start_time": current_time.strftime("%Y-%m-%d %H:%M:%S"),
                "duration_minutes": activity_template["duration_minutes"]
            }

            # Add optional description if venue name is provided (adds context)
            if venue_name:
                activity_entry["venue"] = venue_name

            rehearsal_activities.append(activity_entry)

            # Move to next activity start time
            current_time += timedelta(minutes=activity_template["duration_minutes"])

        # Calculate total duration
        total_duration_minutes = sum(activity["duration_minutes"] for activity in rehearsal_activities)

        # Return the complete rehearsal schedule
        return {
            "rehearsal_activities": rehearsal_activities,
            "total_duration_minutes": total_duration_minutes
        }

    @is_tool()
    def create_budget_category(self, wedding_id: str, category_name: Literal["venue", "catering", "photography", "flowers", "music", "attire", "decorations", "invitations", "transportation", "honeymoon", "gifts", "miscellaneous"], allocated_amount: float, notes: str = None):
        """
        Create a budget category with allocated amount for wedding expenses.

        This method creates a new budget category for a specific wedding event with an allocated amount.
        The category must be one of the predefined enum values.
        """
        import secrets
        import hashlib
        from datetime import datetime

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

        # Validate category_name is one of the allowed enum values
        # This is enforced by Literal type hint, but we add explicit validation for safety
        allowed_categories = ["venue", "catering", "photography", "flowers", "music", "attire", 
                             "decorations", "invitations", "transportation", "honeymoon", "gifts", "miscellaneous"]
        if category_name not in allowed_categories:
            raise ValueError(f"category_name must be one of {allowed_categories}, got '{category_name}'")

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

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

        # Access the database
        db = self.db

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

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

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

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

        # Create new BudgetCategory instance
        new_category = BudgetCategory(
            category_id=category_id,
            wedding_id=wedding_id,
            category_name=category_name,
            allocated_amount=allocated_amount,
            notes=notes,
            creation_timestamp=creation_timestamp
        )

        # Add the new category to the table
        budget_category_table[category_id] = new_category

        # Update the database with the modified table
        setattr(db, "budget_category", budget_category_table)

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

    @is_tool()
    def create_seating_arrangement(
        self,
        wedding_id: str,
        table_number: str,
        table_capacity: int,
        table_location: str = None,
        table_type: Literal["round", "rectangular", "square", "banquet", "head_table"] = None
    ):
        """
        Create a seating arrangement for the wedding reception with table assignments.

        This method creates a new table entry in the seating_table database with the provided
        details. It generates a unique table_id and records the creation timestamp.

        Args:
            wedding_id: Unique identifier of the wedding event
            table_number: Number or identifier of the table
            table_capacity: Maximum number of guests that can be seated at the table
            table_location: Optional location or section of the table in the venue
            table_type: Optional type or shape of the table (must be one of the enum values)

        Returns:
            dict: Contains table_id and creation_timestamp in yyyy-mm-dd HH:MM:SS format

        Raises:
            ValueError: If required parameters are invalid or wedding_id doesn't exist
        """
        import secrets
        import hashlib
        from datetime import datetime

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

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

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

        # Validate table_type if provided (safety protection for enum parameter)
        valid_table_types = ["round", "rectangular", "square", "banquet", "head_table"]
        if table_type is not None and table_type not in valid_table_types:
            raise ValueError(
                f"table_type must be one of {valid_table_types}, got '{table_type}'"
            )

        # Access the database
        db = self.db

        # Initialize seating_table if it doesn't exist
        if db.seating_table is None:
            db.seating_table = {}

        # Check if a table with the same wedding_id and table_number already exists
        for existing_table in db.seating_table.values():
            if (existing_table.wedding_id == wedding_id and 
                existing_table.table_number == table_number):
                raise ValueError(
                    f"A table with number '{table_number}' already exists for wedding '{wedding_id}'"
                )

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

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

        # Get current timestamp
        creation_time = datetime.now()

        # Create new SeatingTable instance
        new_table = SeatingTable(
            table_id=table_id,
            wedding_id=wedding_id,
            table_number=table_number,
            table_capacity=table_capacity,
            table_location=table_location,
            table_type=table_type,
            creation_timestamp=creation_time
        )

        # Store the new table in the database
        db.seating_table[table_id] = new_table

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

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

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

        # Retrieve wedding event data
        wedding_events = getattr(db, 'wedding_event', None)
        if wedding_events is None:
            raise KeyError(f"Wedding event table not found in database")

        # Check if wedding exists
        if wedding_id not in wedding_events:
            raise KeyError(f"Wedding event with ID '{wedding_id}' not found")

        wedding = wedding_events[wedding_id]

        # Retrieve vendor contracts for this wedding
        vendor_contracts = getattr(db, 'vendor_contract', None)
        if vendor_contracts is None:
            raise KeyError(f"Vendor contract table not found in database")

        # Filter vendor contracts by wedding_id
        wedding_vendors = [
            vendor for vendor in vendor_contracts.values() 
            if vendor.wedding_id == wedding_id
        ]

        # Generate emergency contacts list from vendors
        emergency_contacts = []

        # Add wedding coordinator if exists (planner type vendor)
        for vendor in wedding_vendors:
            # Create contact entry for each vendor with contact information
            if vendor.contact_person and vendor.contact_phone:
                contact_entry = {
                    "name": vendor.contact_person,
                    "role": vendor.vendor_type,
                    "phone": vendor.contact_phone,
                    "company": vendor.vendor_name
                }

                # Add backup phone if available (use vendor_name as identifier for potential backup)
                # Note: Current schema doesn't have backup_phone field, so we set it as None
                # In a real scenario, this would come from additional vendor contact data
                contact_entry["backup_phone"] = None

                emergency_contacts.append(contact_entry)

        # Sort contacts by role priority for emergency situations
        # Priority order: planner, venue coordinator, caterer, photographer, etc.
        role_priority = {
            "planner": 1,
            "caterer": 2,
            "photographer": 3,
            "videographer": 4,
            "florist": 5,
            "decorator": 6,
            "dj": 7,
            "entertainment": 8,
            "baker": 9,
            "transportation": 10
        }

        emergency_contacts.sort(key=lambda x: role_priority.get(x["role"], 99))

        # Generate venue emergency information
        venue_emergency_info = {
            "venue_name": wedding.venue_name,
            # Standard emergency information (in real scenario, this would come from venue data)
            "emergency_exit_locations": "Please refer to venue floor plan for emergency exits",
            "first_aid_location": "Contact venue staff for first aid kit location",
            "venue_emergency_phone": "Contact venue directly for emergency procedures"
        }

        # Return the emergency contact list
        return {
            "emergency_contacts": emergency_contacts,
            "venue_emergency_info": venue_emergency_info
        }

    @is_tool()
    def update_guest_rsvp_status(self, wedding_id: str, guest_id: str, rsvp_status: Literal["attending", "declined", "pending", "maybe"], response_date: str = None):
        """
        Update the RSVP status of a guest in the wedding guest list.

        Args:
            wedding_id: Unique identifier of the wedding event
            guest_id: Unique identifier of the guest
            rsvp_status: RSVP response status (must be one of: attending, declined, pending, maybe)
            response_date: Optional date when the RSVP was received in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Contains success status and updated RSVP status

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

        # Access the database
        db = self.db

        # Get the guest table from database
        guest_table = getattr(db, "guest", None)

        # Check if guest table exists
        if guest_table is None:
            raise KeyError(f"Guest table not found in database")

        # Check if the guest exists in the database
        if guest_id not in guest_table:
            raise KeyError(f"Guest with ID '{guest_id}' not found in database")

        # Retrieve the guest record
        guest = guest_table[guest_id]

        # Verify that the guest belongs to the specified wedding
        if guest.wedding_id != wedding_id:
            raise KeyError(f"Guest '{guest_id}' does not belong to wedding '{wedding_id}'")

        # Update the RSVP status
        guest.rsvp_status = rsvp_status

        # Update the response date if provided
        if response_date is not None:
            # Parse the response_date string to datetime object
            # Support both full datetime format (yyyy-mm-dd HH:MM:SS) and date-only format (yyyy-mm-dd)
            try:
                if len(response_date.strip()) == 10:  # Date only format
                    guest.response_date = datetime.strptime(response_date.strip(), "%Y-%m-%d")
                else:  # Full datetime format
                    guest.response_date = datetime.strptime(response_date.strip(), "%Y-%m-%d %H:%M:%S")
            except ValueError as e:
                raise ValueError(f"Invalid date format for response_date. Expected 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd', got '{response_date}'")
        else:
            # If no response_date provided, use current timestamp
            guest.response_date = datetime.now()

        # Update the guest record in the database
        guest_table[guest_id] = guest
        setattr(db, "guest", guest_table)

        # Return success response with updated status
        return {
            "success": True,
            "updated_status": rsvp_status
        }

    @is_tool()
    def record_expense(
        self,
        wedding_id: str,
        category_id: str,
        expense_description: str,
        amount: float,
        expense_date: str,
        payment_method: Literal["cash", "credit_card", "debit_card", "check", "bank_transfer", "online_payment"] = None
    ):
        """
        Record an actual expense against a budget category.

        This method creates a new expense record and updates the budget tracking for the specified category.
        It validates that the budget category exists before recording the expense.

        Args:
            wedding_id: Unique identifier of the wedding event
            category_id: Unique identifier of the budget category
            expense_description: Description of the expense
            amount: Amount of the expense in dollars
            expense_date: Date when the expense was incurred in yyyy-mm-dd HH:MM:SS format
            payment_method: Method used for payment (optional, must be one of the enum values if provided)

        Returns:
            dict: Contains expense_id and remaining_budget

        Raises:
            ValueError: If budget category doesn't exist, wedding_id mismatch, invalid amount, 
                       invalid date format, or invalid payment method
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access database
        db = self.db

        # Validate amount
        if not isinstance(amount, (int, float)) or amount <= 0:
            raise ValueError(f"Invalid amount: {amount}. Amount must be a positive number.")

        # Parse and validate expense_date format
        try:
            parsed_expense_date = datetime.strptime(expense_date, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            try:
                # Try date-only format as fallback
                parsed_expense_date = datetime.strptime(expense_date, "%Y-%m-%d")
            except ValueError:
                raise ValueError(f"Invalid date format: {expense_date}. Expected format: yyyy-mm-dd HH:MM:SS or yyyy-mm-dd")

        # Validate payment_method if provided (safety protection for enum parameter)
        valid_payment_methods = ["cash", "credit_card", "debit_card", "check", "bank_transfer", "online_payment"]
        if payment_method is not None and payment_method not in valid_payment_methods:
            raise ValueError(
                f"Invalid payment_method: {payment_method}. "
                f"Must be one of: {', '.join(valid_payment_methods)}"
            )

        # Get budget_category table
        budget_category_table = getattr(db, "budget_category", None)
        if budget_category_table is None:
            raise ValueError("Budget category table does not exist in database.")

        # Check if the budget category exists
        budget_category = budget_category_table.get(category_id)
        if budget_category is None:
            raise ValueError(f"Budget category with ID '{category_id}' does not exist.")

        # Verify that the budget category belongs to the specified wedding
        if budget_category.wedding_id != wedding_id:
            raise ValueError(
                f"Budget category '{category_id}' does not belong to wedding '{wedding_id}'. "
                f"It belongs to wedding '{budget_category.wedding_id}'."
            )

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

        # Import Expense class (already imported at file header, can use directly)
        from datetime import datetime

        # Create new expense record
        new_expense = Expense(
            expense_id=expense_id,
            wedding_id=wedding_id,
            category_id=category_id,
            expense_description=expense_description,
            amount=amount,
            expense_date=parsed_expense_date,
            payment_method=payment_method
        )

        # Get expense table
        expense_table = getattr(db, "expense", None)
        if expense_table is None:
            # Initialize expense table if it doesn't exist
            expense_table = {}

        # Add new expense to the table
        expense_table[expense_id] = new_expense
        setattr(db, "expense", expense_table)

        # Calculate total expenses for this category
        total_expenses = 0.0
        for exp_id, exp in expense_table.items():
            if exp.category_id == category_id and exp.wedding_id == wedding_id:
                total_expenses += exp.amount

        # Calculate remaining budget
        remaining_budget = budget_category.allocated_amount - total_expenses

        # Return expense_id and remaining_budget
        return {
            "expense_id": expense_id,
            "remaining_budget": remaining_budget
        }

    @is_tool()
    def assign_guest_to_table(
        self,
        wedding_id: str,
        guest_id: str,
        table_id: str,
        seat_number: Optional[int] = None
    ) -> dict:
        """
        Assign a guest to a specific table in the seating arrangement.

        This method creates or updates a seating assignment for a guest at a specified table.
        It validates that the table exists and has sufficient capacity before making the assignment.

        Args:
            wedding_id: Unique identifier of the wedding event
            guest_id: Unique identifier of the guest to be assigned
            table_id: Unique identifier of the table where guest will be seated
            seat_number: Optional specific seat number at the table

        Returns:
            dict: Contains 'success' (bool) and 'remaining_capacity' (int)

        Raises:
            ValueError: If wedding_id, guest_id, or table_id is empty
            ValueError: If table does not exist
            ValueError: If table has no remaining capacity
            ValueError: If seat_number is provided and already occupied
        """
        import secrets
        import hashlib

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

        # Access database
        db = self.db

        # Get seating_table and guest_seating_assignment data
        seating_tables = getattr(db, "seating_table", None)
        guest_assignments = getattr(db, "guest_seating_assignment", None)

        # Initialize tables if they don't exist
        if seating_tables is None:
            seating_tables = {}
            setattr(db, "seating_table", seating_tables)

        if guest_assignments is None:
            guest_assignments = {}
            setattr(db, "guest_seating_assignment", guest_assignments)

        # Validate that the table exists and belongs to the specified wedding
        target_table = None
        for table in seating_tables.values():
            if table.table_id == table_id and table.wedding_id == wedding_id:
                target_table = table
                break

        if target_table is None:
            raise ValueError(f"Table with table_id '{table_id}' does not exist for wedding '{wedding_id}'")

        # Count current assignments for this table
        current_assignments = [
            assignment for assignment in guest_assignments.values()
            if assignment.table_id == table_id and assignment.wedding_id == wedding_id
        ]

        current_count = len(current_assignments)

        # Check if table has capacity
        if current_count >= target_table.table_capacity:
            raise ValueError(
                f"Table '{table_id}' has no remaining capacity "
                f"(current: {current_count}, max: {target_table.table_capacity})"
            )

        # If seat_number is provided, check if it's already occupied
        if seat_number is not None:
            for assignment in current_assignments:
                if assignment.seat_number == seat_number:
                    raise ValueError(
                        f"Seat number {seat_number} at table '{table_id}' is already occupied"
                    )

        # Check if guest already has an assignment for this wedding
        existing_assignment = None
        for assignment_id, assignment in guest_assignments.items():
            if assignment.guest_id == guest_id and assignment.wedding_id == wedding_id:
                existing_assignment = assignment_id
                break

        # Create or update the assignment
        if existing_assignment:
            # Update existing assignment
            assignment = guest_assignments[existing_assignment]

            # If moving from another table, we need to recalculate
            old_table_id = assignment.table_id
            assignment.table_id = table_id
            assignment.seat_number = seat_number

            # Update in database
            setattr(db, "guest_seating_assignment", guest_assignments)

            # Recalculate current count if table changed
            if old_table_id != table_id:
                current_assignments = [
                    a for a in guest_assignments.values()
                    if a.table_id == table_id and a.wedding_id == wedding_id
                ]
                current_count = len(current_assignments)
        else:
            # Create new assignment with generated ID
            assignment_id = "ASSIGN_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            new_assignment = GuestSeatingAssignment(
                assignment_id=assignment_id,
                wedding_id=wedding_id,
                guest_id=guest_id,
                table_id=table_id,
                seat_number=seat_number
            )

            guest_assignments[assignment_id] = new_assignment

            # Update database
            setattr(db, "guest_seating_assignment", guest_assignments)

            # Increment current count for new assignment
            current_count += 1

        # Calculate remaining capacity
        remaining_capacity = target_table.table_capacity - current_count

        return {
            "success": True,
            "remaining_capacity": remaining_capacity
        }

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

        # Retrieve invitation and guest tables from database
        invitation_table = getattr(db, "invitation", None)
        guest_table = getattr(db, "guest", None)

        # Validate that required tables exist
        if invitation_table is None:
            raise KeyError("Invitation table not found in database")
        if guest_table is None:
            raise KeyError("Guest table not found in database")

        # Filter invitations for the specified wedding_id
        wedding_invitations = {
            inv_id: inv for inv_id, inv in invitation_table.items()
            if inv.wedding_id == wedding_id
        }

        # Check if wedding exists (has invitations)
        if not wedding_invitations:
            raise KeyError(f"No invitations found for wedding_id: {wedding_id}")

        # Initialize counters
        total_invitations_sent = len(wedding_invitations)
        confirmed_count = 0
        declined_count = 0
        pending_count = 0

        # Get all guest_ids from invitations for this wedding
        invited_guest_ids = {inv.guest_id for inv in wedding_invitations.values()}

        # Count responses by checking guest RSVP status
        for guest_id in invited_guest_ids:
            guest = guest_table.get(guest_id)

            # If guest not found, treat as pending
            if guest is None:
                pending_count += 1
                continue

            # Verify guest belongs to the same wedding
            if guest.wedding_id != wedding_id:
                continue

            # Count based on RSVP status
            if guest.rsvp_status == "attending":
                confirmed_count += 1
            elif guest.rsvp_status == "declined":
                declined_count += 1
            elif guest.rsvp_status in ["pending", "maybe"]:
                # Treat "maybe" as pending since they haven't confirmed
                pending_count += 1

        # Calculate responses received (confirmed + declined)
        responses_received = confirmed_count + declined_count

        # Calculate response rate as percentage
        # Response rate = (responses_received / total_invitations_sent) * 100
        if total_invitations_sent > 0:
            response_rate = round((responses_received / total_invitations_sent) * 100, 2)
        else:
            response_rate = 0.0

        # Return statistics dictionary
        return {
            "total_invitations_sent": total_invitations_sent,
            "responses_received": responses_received,
            "confirmed_count": confirmed_count,
            "declined_count": declined_count,
            "pending_count": pending_count,
            "response_rate": response_rate
        }

    @is_tool()
    def create_wedding_website(
        self,
        wedding_id: str,
        website_url: str,
        theme_name: Literal["classic", "modern", "rustic", "elegant", "beach", "garden", "vintage"] = None,
        welcome_message: str = None,
        include_rsvp_form: bool = None,
        include_registry_link: bool = None
    ):
        """
        Create a wedding website with details and customization options.

        This method creates a new wedding website entry in the database with the provided
        configuration and customization options. It validates the theme_name if provided
        and generates a unique website_id.

        Args:
            wedding_id: Unique identifier of the wedding event
            website_url: Custom URL for the wedding website
            theme_name: Theme or design style for the website (optional, must be one of the enum values)
            welcome_message: Welcome message displayed on the website (optional)
            include_rsvp_form: Whether to include an RSVP form on the website (optional)
            include_registry_link: Whether to include a link to the gift registry (optional)

        Returns:
            dict: Dictionary containing:
                - website_id: Unique identifier for the created wedding website
                - website_url: URL of the created wedding website
                - creation_timestamp: Timestamp when the website was created in yyyy-mm-dd HH:MM:SS format

        Raises:
            ValueError: If wedding_id or website_url is empty/invalid, or if theme_name is not in allowed values
        """
        import secrets
        import hashlib
        from datetime import datetime

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

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

        # Validate theme_name if provided (enum constraint with safety protection)
        allowed_themes = ["classic", "modern", "rustic", "elegant", "beach", "garden", "vintage"]
        if theme_name is not None:
            if not isinstance(theme_name, str):
                raise ValueError("theme_name must be a string")
            if theme_name not in allowed_themes:
                raise ValueError(
                    f"theme_name must be one of {allowed_themes}, got '{theme_name}'"
                )

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

        if include_registry_link is not None and not isinstance(include_registry_link, bool):
            raise ValueError("include_registry_link must be a boolean value")

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

        # Get current timestamp
        creation_timestamp = datetime.now()

        # Access database
        db = self.db

        # Initialize wedding_website table if it doesn't exist
        if db.wedding_website is None:
            db.wedding_website = {}

        # Create new WeddingWebsite instance
        new_website = WeddingWebsite(
            website_id=website_id,
            wedding_id=wedding_id,
            website_url=website_url,
            theme_name=theme_name,
            welcome_message=welcome_message,
            include_rsvp_form=include_rsvp_form,
            include_registry_link=include_registry_link,
            creation_timestamp=creation_timestamp
        )

        # Store the new website in the database
        db.wedding_website[website_id] = new_website

        # Update the database
        setattr(db, "wedding_website", db.wedding_website)

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

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

    @is_tool()
    def generate_gift_registry(
        self,
        wedding_id: str,
        registry_name: str,
        registry_url: Optional[str] = None,
        store_name: Optional[str] = None
    ) -> dict:
        """
        Generate a gift registry with items and their details.

        Args:
            wedding_id: Unique identifier of the wedding event
            registry_name: Name of the gift registry
            registry_url: Optional URL link to the online registry
            store_name: Optional name of the store hosting the registry

        Returns:
            dict: Contains registry_id and creation_timestamp

        Raises:
            ValueError: If wedding_id or registry_name is empty/invalid
        """
        import secrets
        import hashlib
        from datetime import datetime

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

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

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

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

        # Access the database
        db = self.db

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

        # Generate unique registry_id using secure random token
        prefix = "REGISTRY"
        registry_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10].upper()

        # Ensure registry_id is unique (collision check)
        gift_registry_table = getattr(db, "gift_registry")
        while registry_id in gift_registry_table:
            registry_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10].upper()

        # Get current timestamp
        creation_timestamp = datetime.now()

        # Create new GiftRegistry instance
        new_registry = GiftRegistry(
            registry_id=registry_id,
            wedding_id=wedding_id.strip(),
            registry_name=registry_name.strip(),
            registry_url=registry_url.strip() if registry_url else None,
            store_name=store_name.strip() if store_name else None,
            creation_timestamp=creation_timestamp
        )

        # Add the new registry to the database
        gift_registry_table[registry_id] = new_registry
        setattr(db, "gift_registry", gift_registry_table)

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

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

    @is_tool()
    def get_budget_summary(self, wedding_id: str):
        # Import required datetime module for time handling
        from datetime import datetime

        # Get database instance
        db = self.db

        # Retrieve budget_category and expense tables from database
        budget_categories = getattr(db, "budget_category", None)
        expenses = getattr(db, "expense", None)

        # Validate that required tables exist in database
        if budget_categories is None:
            raise KeyError("budget_category table not found in database")
        if expenses is None:
            raise KeyError("expense table not found in database")

        # Filter budget categories for the specified wedding
        wedding_categories = {}
        for cat_id, category in budget_categories.items():
            if category.wedding_id == wedding_id:
                wedding_categories[cat_id] = category

        # Check if wedding exists by verifying if any budget categories are found
        if not wedding_categories:
            raise KeyError(f"Wedding with id '{wedding_id}' not found or has no budget categories")

        # Initialize total budget by summing all allocated amounts across categories
        total_budget = sum(cat.allocated_amount for cat in wedding_categories.values())

        # Calculate total spent by summing all expenses for this wedding
        total_spent = 0.0
        # Dictionary to track spending per category for detailed breakdown
        category_spending = {cat_id: 0.0 for cat_id in wedding_categories.keys()}

        # Iterate through all expenses to calculate spending
        for expense in expenses.values():
            # Only process expenses that belong to this wedding
            if expense.wedding_id == wedding_id:
                total_spent += expense.amount
                # Add expense amount to corresponding category if category exists
                if expense.category_id in category_spending:
                    category_spending[expense.category_id] += expense.amount

        # Calculate remaining budget
        remaining_budget = total_budget - total_spent

        # Build detailed category breakdown list
        categories = []
        for cat_id, category in wedding_categories.items():
            # Get total spent for this category from tracking dictionary
            spent_in_category = category_spending.get(cat_id, 0.0)

            # Create category summary object with all required fields
            category_summary = {
                "category_name": category.category_name,
                "allocated": category.allocated_amount,
                "spent": spent_in_category
            }
            categories.append(category_summary)

        # Sort categories by name for consistent output
        categories.sort(key=lambda x: x["category_name"])

        # Return complete budget summary with all required fields
        return {
            "total_budget": total_budget,
            "total_spent": total_spent,
            "remaining_budget": remaining_budget,
            "categories": categories
        }

    @is_tool()
    def add_gift_item(
        self,
        wedding_id: str,
        registry_id: str,
        item_name: str,
        price: float,
        quantity_desired: int,
        item_description: str = None,
        category: Literal["kitchen", "bedroom", "bathroom", "dining", "home_decor", "electronics", "outdoor", "experience", "cash_fund"] = None
    ):
        """
        Add a gift item to the wedding registry.

        This method creates a new gift item entry in the registry with the specified details.
        It generates a unique item_id and records the creation timestamp.

        Args:
            wedding_id: Unique identifier of the wedding event
            registry_id: Unique identifier of the gift registry
            item_name: Name of the gift item
            price: Price of the item in dollars
            quantity_desired: Number of this item desired
            item_description: Optional description of the gift item
            category: Optional category of the gift item (must be one of the enum values if provided)

        Returns:
            dict: Contains item_id and creation_timestamp in "yyyy-mm-dd HH:MM:SS" format

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

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

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

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

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

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

        # Validate category if provided - enum validation
        valid_categories = [
            "kitchen", "bedroom", "bathroom", "dining", "home_decor",
            "electronics", "outdoor", "experience", "cash_fund"
        ]
        if category is not None and category not in valid_categories:
            raise ValueError(f"category must be one of {valid_categories} if provided")

        # Access the database
        db = self.db

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

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

        # Ensure uniqueness of item_id
        gift_item_table = getattr(db, "gift_item")
        while item_id in gift_item_table:
            item_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        creation_timestamp = datetime.now()

        # Create new GiftItem instance
        new_item = GiftItem(
            item_id=item_id,
            wedding_id=wedding_id,
            registry_id=registry_id,
            item_name=item_name,
            item_description=item_description,
            price=price,
            quantity_desired=quantity_desired,
            quantity_purchased=0,  # Initially no items purchased
            category=category,
            purchased_by=None,  # No purchaser initially
            purchase_date=None,  # No purchase date initially
            creation_timestamp=creation_timestamp
        )

        # Add the new item to the database
        gift_item_table[item_id] = new_item
        setattr(db, "gift_item", gift_item_table)

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

    @is_tool()
    def generate_vendor_payment_schedule(self, wedding_id: str):
        """
        Generate a payment schedule for all vendors based on contract terms.

        This method retrieves all vendor contracts for a specific wedding and generates
        a comprehensive payment schedule showing upcoming payments, including final payments
        and any outstanding balances.

        Args:
            wedding_id: Unique identifier of the wedding event

        Returns:
            dict: Contains payment_schedule (list of payment milestones) and 
                  total_upcoming_payments (total amount due)

        Raises:
            KeyError: If wedding_id does not exist or no vendor contracts found
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get vendor_contract table
        vendor_contract_table = getattr(db, 'vendor_contract', None)

        # Validate that vendor_contract table exists
        if vendor_contract_table is None:
            raise KeyError(f"vendor_contract table not found in database")

        # Find all vendor contracts for the specified wedding
        wedding_contracts = []
        for contract_id, contract in vendor_contract_table.items():
            if contract.wedding_id == wedding_id:
                wedding_contracts.append(contract)

        # Raise error if no contracts found for this wedding
        if not wedding_contracts:
            raise KeyError(f"No vendor contracts found for wedding_id: {wedding_id}")

        # Generate payment schedule
        payment_schedule = []
        total_upcoming_payments = 0.0

        # Get current datetime for comparison
        current_time = datetime.now()

        for contract in wedding_contracts:
            # Calculate remaining balance
            remaining_balance = contract.contract_amount - contract.amount_paid

            # Only include contracts with remaining balance
            if remaining_balance > 0:
                # Determine payment date and type based on contract status and service date
                payment_date = None
                payment_type = None
                amount = remaining_balance

                if contract.payment_status == "not_paid":
                    # If not paid at all, check if deposit is required
                    if contract.deposit_amount and contract.deposit_amount > 0:
                        # Next payment is deposit
                        payment_type = "deposit_payment"
                        amount = contract.deposit_amount
                        # Set payment date to 30 days before service date if service_date exists
                        if contract.service_date:
                            payment_date = contract.service_date
                        else:
                            # If no service date, use current time
                            payment_date = current_time
                    else:
                        # No deposit required, full payment needed
                        payment_type = "full_payment"
                        amount = remaining_balance
                        payment_date = contract.service_date if contract.service_date else current_time

                elif contract.payment_status == "deposit_paid":
                    # Deposit already paid, remaining is final payment
                    payment_type = "final_payment"
                    amount = remaining_balance
                    # Final payment typically due on or before service date
                    payment_date = contract.service_date if contract.service_date else current_time

                elif contract.payment_status == "partially_paid":
                    # Partially paid, remaining is final payment
                    payment_type = "final_payment"
                    amount = remaining_balance
                    payment_date = contract.service_date if contract.service_date else current_time

                elif contract.payment_status == "overdue":
                    # Overdue payment, mark as urgent
                    payment_type = "overdue_payment"
                    amount = remaining_balance
                    # Use service date if available, otherwise current time
                    payment_date = contract.service_date if contract.service_date else current_time

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

                # Create payment milestone entry
                payment_milestone = {
                    "vendor_name": contract.vendor_name,
                    "payment_date": payment_date_str,
                    "amount": round(amount, 2),
                    "payment_type": payment_type
                }

                payment_schedule.append(payment_milestone)
                total_upcoming_payments += amount

        # Sort payment schedule by payment_date (earliest first)
        payment_schedule.sort(key=lambda x: datetime.strptime(x["payment_date"], "%Y-%m-%d %H:%M:%S"))

        # Round total to 2 decimal places
        total_upcoming_payments = round(total_upcoming_payments, 2)

        return {
            "payment_schedule": payment_schedule,
            "total_upcoming_payments": total_upcoming_payments
        }

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

        # Retrieve guest and gift_item tables from database
        guests = getattr(db, "guest", None)
        gift_items = getattr(db, "gift_item", None)

        # Validate that required tables exist
        if guests is None:
            raise KeyError("Guest table not found in database")
        if gift_items is None:
            raise KeyError("Gift item table not found in database")

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

        # Filter gift items for the specified wedding that have been purchased
        # A gift is considered received if it has a purchaser and purchase date
        purchased_gifts = []
        for item_id, gift_item in gift_items.items():
            if (gift_item.wedding_id == wedding_id and 
                gift_item.purchased_by is not None and 
                gift_item.purchase_date is not None):
                purchased_gifts.append(gift_item)

        # Build a mapping from guest names to their gift information
        # Use fuzzy matching to handle variations in guest names
        from thefuzz import fuzz, process

        # Create a list of all guest names for the wedding
        wedding_guest_names = []
        guest_name_to_guest = {}
        for guest_id, guest in guests.items():
            if guest.wedding_id == wedding_id:
                wedding_guest_names.append(guest.guest_name)
                guest_name_to_guest[guest.guest_name] = guest

        # Dictionary to track thank you note information per guest
        # Key: matched guest name, Value: list of gift details
        guest_thank_you_info = {}

        # Process each purchased gift and match to guests
        for gift_item in purchased_gifts:
            purchaser_name = gift_item.purchased_by

            # Use fuzzy matching to find the best matching guest
            # This handles variations in name formatting
            if wedding_guest_names:
                match_result = process.extractOne(purchaser_name, wedding_guest_names)
                if match_result:
                    matched_name, similarity_score = match_result
                    # Use a threshold of 80 for name matching
                    # Names with 80+ similarity are considered the same person
                    if similarity_score >= 80:
                        # Initialize guest entry if not exists
                        if matched_name not in guest_thank_you_info:
                            guest_thank_you_info[matched_name] = {
                                "gifts": [],
                                "thank_you_sent": False  # Assume not sent initially
                            }

                        # Add gift information
                        guest_thank_you_info[matched_name]["gifts"].append({
                            "gift_description": gift_item.item_name,
                            "purchase_date": gift_item.purchase_date
                        })

        # Build the thank you list output
        thank_you_list = []
        notes_sent = 0

        for guest_name, info in guest_thank_you_info.items():
            # For each guest, create entries for each gift they gave
            # This allows tracking individual thank you notes per gift
            for gift in info["gifts"]:
                thank_you_entry = {
                    "guest_name": guest_name,
                    "gift_description": gift["gift_description"],
                    "thank_you_sent": info["thank_you_sent"]
                }
                thank_you_list.append(thank_you_entry)

                # Count how many thank you notes have been sent
                if info["thank_you_sent"]:
                    notes_sent += 1

        # Calculate total number of thank you notes needed
        total_thank_you_notes_needed = len(thank_you_list)

        # Return the complete thank you list with statistics
        return {
            "thank_you_list": thank_you_list,
            "total_thank_you_notes_needed": total_thank_you_notes_needed,
            "notes_sent": notes_sent
        }

    @is_tool()
    def get_seating_chart(self, wedding_id: str):
        # Access the database
        db = self.db

        # Retrieve all tables, assignments, and guests from database
        seating_tables = getattr(db, "seating_table", None)
        guest_assignments = getattr(db, "guest_seating_assignment", None)
        guests = getattr(db, "guest", None)

        # Validate that wedding exists by checking if any table exists for this wedding
        # This serves as a proxy check since we don't have a direct wedding table
        wedding_exists = False
        if seating_tables:
            for table in seating_tables.values():
                if table.wedding_id == wedding_id:
                    wedding_exists = True
                    break

        # If no tables found for this wedding, check if any assignments or guests exist
        if not wedding_exists:
            if guest_assignments:
                for assignment in guest_assignments.values():
                    if assignment.wedding_id == wedding_id:
                        wedding_exists = True
                        break

            if not wedding_exists and guests:
                for guest in guests.values():
                    if guest.wedding_id == wedding_id:
                        wedding_exists = True
                        break

        # Raise KeyError if wedding doesn't exist
        if not wedding_exists:
            raise KeyError(f"Wedding with ID '{wedding_id}' does not exist")

        # Initialize result structure
        tables_list = []
        total_seated_guests = 0

        # Build a mapping of table_id to guest names for efficient lookup
        table_to_guests = {}

        if guest_assignments and guests:
            # Iterate through all guest assignments for this wedding
            for assignment in guest_assignments.values():
                if assignment.wedding_id == wedding_id:
                    # Look up the guest name using guest_id
                    guest = guests.get(assignment.guest_id)
                    if guest:
                        # Add guest name to the corresponding table
                        if assignment.table_id not in table_to_guests:
                            table_to_guests[assignment.table_id] = []
                        table_to_guests[assignment.table_id].append(guest.guest_name)
                        total_seated_guests += 1

        # Process all tables for this wedding
        if seating_tables:
            for table in seating_tables.values():
                if table.wedding_id == wedding_id:
                    # Get the list of guests assigned to this table
                    assigned_guests = table_to_guests.get(table.table_id, [])

                    # Create table entry with assigned guests
                    table_entry = {
                        "table_id": table.table_id,
                        "table_number": table.table_number,
                        "guests": assigned_guests
                    }

                    tables_list.append(table_entry)

        # Sort tables by table_number for consistent ordering
        # Extract numeric part if table_number follows "Table X" pattern
        # Fixed: Always return a tuple (int, str) for consistent type comparison
        def get_sort_key(table_entry):
            table_num = table_entry["table_number"]
            # Try to extract number from strings like "Table 5"
            try:
                if "table" in table_num.lower():
                    parts = table_num.split()
                    for part in parts:
                        if part.isdigit():
                            # Return tuple: (numeric_value, original_string) for consistent sorting
                            return (int(part), table_num)
                # If it's just a number string
                if table_num.isdigit():
                    return (int(table_num), table_num)
            except:
                pass
            # Fallback: return tuple with large number to sort non-numeric entries last
            return (float('inf'), table_num)

        tables_list.sort(key=get_sort_key)

        # Compile final result
        result = {
            "tables": tables_list,
            "total_tables": len(tables_list),
            "total_seated_guests": total_seated_guests
        }

        return result
