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

class EmailManagementTools(ToolKitBase):
    """All tools for email_management."""
    
    db: EmailManagementDB
    
    def __init__(self, db: EmailManagementDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def calculate_email_size(self, body_length: int, attachment_sizes: list = None) -> dict:
        """
        Calculate the total size of an email including attachments.

        This method computes the total size of an email by considering:
        1. The body length in characters (converted to bytes assuming UTF-8 encoding)
        2. The sizes of all attachments (if provided)

        Args:
            body_length: Length of email body in characters
            attachment_sizes: Optional list of attachment sizes in bytes

        Returns:
            Dictionary containing:
            - total_size_bytes: Total size in bytes
            - total_size_mb: Total size in megabytes (rounded to 2 decimal places)

        Raises:
            ValueError: If body_length is negative or if any attachment size is negative
            TypeError: If body_length is not an integer or attachment_sizes contains non-integer values
        """
        # Validate body_length parameter
        if not isinstance(body_length, int):
            raise TypeError(f"body_length must be an integer, got {type(body_length).__name__}")

        if body_length < 0:
            raise ValueError(f"body_length must be non-negative, got {body_length}")

        # Calculate body size in bytes
        # Assuming UTF-8 encoding where ASCII characters = 1 byte, but allowing for multi-byte characters
        # For a more accurate estimation, we assume average of 1.5 bytes per character for mixed content
        # However, for simplicity and common practice, we'll use 1 byte per character as baseline
        body_size_bytes = body_length

        # Initialize total size with body size
        total_size_bytes = body_size_bytes

        # Process attachment sizes if provided
        if attachment_sizes is not None:
            # Validate attachment_sizes is a list or iterable
            if not isinstance(attachment_sizes, (list, tuple)):
                raise TypeError(f"attachment_sizes must be a list or tuple, got {type(attachment_sizes).__name__}")

            # Validate and sum all attachment sizes
            for idx, size in enumerate(attachment_sizes):
                if not isinstance(size, int):
                    raise TypeError(f"attachment_sizes[{idx}] must be an integer, got {type(size).__name__}")

                if size < 0:
                    raise ValueError(f"attachment_sizes[{idx}] must be non-negative, got {size}")

                total_size_bytes += size

        # Convert bytes to megabytes (1 MB = 1024 * 1024 bytes = 1048576 bytes)
        total_size_mb = round(total_size_bytes / 1048576, 2)

        # Return the calculated sizes
        return {
            "total_size_bytes": total_size_bytes,
            "total_size_mb": total_size_mb
        }

    @is_tool()
    def add_flag_to_email(self, message_id: str):
        """
        Add a flag or star marker to an email for follow-up

        Args:
            message_id: Unique identifier of the email message

        Returns:
            dict: Contains 'success' boolean indicating whether the flag was added successfully

        Raises:
            KeyError: If the email message does not exist in the mailbox
        """
        # Access the database instance
        db = self.db

        # Retrieve the email_message table from the database
        email_message_table = getattr(db, "email_message", None)

        # Check if the email_message table exists
        if email_message_table is None:
            raise KeyError(f"Email message table does not exist in the database")

        # Check if the message_id exists in the email_message table
        if message_id not in email_message_table:
            raise KeyError(f"Email message with ID '{message_id}' does not exist in the mailbox")

        # Retrieve the email message object
        email_message = email_message_table[message_id]

        # Set the is_flagged attribute to True to mark the email as flagged
        email_message.is_flagged = True

        # Update the email message in the database
        # Since we're modifying the object in place, we need to update the table
        email_message_table[message_id] = email_message
        setattr(db, "email_message", email_message_table)

        # Return success response
        return {
            "success": True
        }

    @is_tool()
    def create_email_folder(self, folder_name: str, parent_folder_id: str = None) -> dict:
        """
        Create a new folder for organizing emails.

        This method creates a new email folder with a unique identifier and adds it to the database.
        If a parent_folder_id is provided, the folder will be nested under that parent folder.

        Args:
            folder_name: Name of the new folder to create
            parent_folder_id: Optional identifier of parent folder for nested structure

        Returns:
            dict: Dictionary containing the folder_id of the created folder

        Raises:
            ValueError: If folder_name is empty, or if parent_folder_id is provided but doesn't exist,
                       or if a folder with the same name already exists under the same parent
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

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

        folder_name = folder_name.strip()

        # Get existing email folders from database
        email_folders = getattr(db, 'email_folder', None)
        if email_folders is None:
            email_folders = {}

        # If parent_folder_id is provided, validate it exists
        if parent_folder_id is not None:
            if parent_folder_id not in email_folders:
                raise ValueError(f"Parent folder with id '{parent_folder_id}' does not exist")

        # Check if a folder with the same name already exists under the same parent
        # This ensures folder names are unique within the same parent scope
        for existing_folder in email_folders.values():
            if (existing_folder.folder_name == folder_name and 
                existing_folder.parent_folder_id == parent_folder_id):
                raise ValueError(
                    f"A folder named '{folder_name}' already exists under the same parent"
                )

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

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

        # Create the new EmailFolder instance
        new_folder = EmailFolder(
            folder_id=folder_id,
            folder_name=folder_name,
            parent_folder_id=parent_folder_id,
            is_system_folder=False,  # User-created folders are not system folders
            created_at=datetime.now()
        )

        # Add the new folder to the database
        email_folders[folder_id] = new_folder
        setattr(db, 'email_folder', email_folders)

        # Return the folder_id of the created folder
        return {
            "folder_id": folder_id
        }

    @is_tool()
    def retrieve_calendar_event(self, event_id: str):
        """
        Get complete details of a specific calendar event

        This method retrieves a calendar event from the database using its unique identifier
        and returns all relevant details including title, time, and attendees.

        Args:
            event_id: Unique identifier of the event to retrieve

        Returns:
            dict: Dictionary containing event details with keys:
                - event_id: Unique identifier of the event
                - title: Title of the event
                - start_time: Start time in yyyy-mm-dd HH:MM:SS format
                - end_time: End time in yyyy-mm-dd HH:MM:SS format
                - attendees: List of attendee email addresses

        Raises:
            KeyError: If the event_id does not exist in the calendar
        """
        from datetime import datetime

        # Get the database instance
        db = self.db

        # Retrieve the calendar_event table from the database
        calendar_event_table = getattr(db, "calendar_event", None)

        # Check if the table exists
        if calendar_event_table is None:
            raise KeyError(f"Calendar event table not found in database")

        # Attempt to retrieve the event by event_id
        # The table is a Dict[str, CalendarEvent] where key is event_id
        event = calendar_event_table.get(event_id)

        # If event doesn't exist, raise KeyError
        if event is None:
            raise KeyError(f"Event with id '{event_id}' not found in calendar")

        # Format datetime objects to string format "yyyy-mm-dd HH:MM:SS"
        # Handle the case where start_time or end_time might be None (though schema requires them)
        start_time_str = event.start_time.strftime("%Y-%m-%d %H:%M:%S") if event.start_time else None
        end_time_str = event.end_time.strftime("%Y-%m-%d %H:%M:%S") if event.end_time else None

        # Prepare the return dictionary with required fields
        result = {
            "event_id": event.event_id,
            "title": event.title,
            "start_time": start_time_str,
            "end_time": end_time_str,
            "attendees": event.attendees if event.attendees is not None else []
        }

        return result

    @is_tool()
    def create_custom_label(self, label_name: str, color: Optional[str] = None) -> dict:
        """
        Create a new custom label for email categorization.

        This method creates a new EmailLabel entry in the database with the provided
        label name and optional color code. It validates that the label name is unique
        and generates a unique label_id.

        Args:
            label_name: Name of the new label (must be unique and non-empty)
            color: Optional color code for the label display (e.g., "#FF5733")

        Returns:
            dict: Contains the label_id of the newly created label
                {
                    "label_id": "label_xxx"
                }

        Raises:
            ValueError: If label_name is empty, already exists, or color format is invalid
        """
        import secrets
        import hashlib
        import re
        from datetime import datetime

        # Access the database
        db = self.db

        # Validate label_name is not empty or whitespace only
        if not label_name or not label_name.strip():
            raise ValueError("Label name cannot be empty or whitespace only")

        # Normalize label_name by stripping whitespace
        label_name = label_name.strip()

        # Get existing email_label table data
        email_label_table = getattr(db, "email_label", None)
        if email_label_table is None:
            # Initialize empty table if it doesn't exist
            email_label_table = {}

        # Check if label_name already exists (case-insensitive comparison for better UX)
        for existing_label in email_label_table.values():
            if existing_label.label_name.lower() == label_name.lower():
                raise ValueError(f"Label name '{label_name}' already exists")

        # Validate color format if provided (should be hex color code)
        if color is not None:
            color = color.strip()
            # Check if color matches hex color format (#RRGGBB or #RGB)
            if not re.match(r'^#(?:[0-9a-fA-F]{3}){1,2}$', color):
                raise ValueError(f"Invalid color format '{color}'. Expected hex color code (e.g., #FF5733 or #F73)")

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

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

        # Create new EmailLabel instance
        new_label = EmailLabel(
            label_id=label_id,
            label_name=label_name,
            color=color,
            is_system_label=False,  # Custom labels are not system labels
            created_at=datetime.now()
        )

        # Add the new label to the table
        email_label_table[label_id] = new_label

        # Write back to database
        setattr(db, "email_label", email_label_table)

        # Return the label_id of the newly created label
        return {
            "label_id": label_id
        }

    @is_tool()
    def move_email_to_folder(self, message_id: str, folder_id: str):
        """
        Move an email from current folder to a specified destination folder.

        Args:
            message_id: Unique identifier of the email message
            folder_id: Identifier of the destination folder

        Returns:
            dict: {"success": bool} indicating whether the move operation succeeded

        Raises:
            KeyError: If email message or destination folder does not exist
        """
        # Get database instance
        db = self.db

        # Retrieve email_message table from database
        email_message_table = getattr(db, "email_message", None)
        if email_message_table is None:
            raise KeyError("email_message table does not exist in database")

        # Retrieve email_folder table from database
        email_folder_table = getattr(db, "email_folder", None)
        if email_folder_table is None:
            raise KeyError("email_folder table does not exist in database")

        # Check if the email message exists
        if message_id not in email_message_table:
            raise KeyError(f"Email message with message_id '{message_id}' does not exist")

        # Check if the destination folder exists
        if folder_id not in email_folder_table:
            raise KeyError(f"Email folder with folder_id '{folder_id}' does not exist")

        # Get the email message object
        email_message = email_message_table[message_id]

        # Update the folder_id of the email message to the destination folder
        email_message.folder_id = folder_id

        # Write the updated email message back to the database
        email_message_table[message_id] = email_message
        setattr(db, "email_message", email_message_table)

        # Return success result
        return {"success": True}

    @is_tool()
    def get_unread_email_count(self, folder_id: str):
        """
        Get the count of unread emails in a specific folder.

        This method retrieves all emails in the specified folder and counts
        how many of them are unread (is_read == False).

        Args:
            folder_id: Identifier of the folder to check

        Returns:
            dict: Contains 'unread_count' - the number of unread emails

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

        # Get the email_folder table to verify folder exists
        email_folder_table = getattr(db, "email_folder", None)
        if email_folder_table is None:
            raise KeyError(f"email_folder table does not exist in database")

        # Check if the folder exists in the database
        if folder_id not in email_folder_table:
            raise KeyError(f"Folder with folder_id '{folder_id}' does not exist")

        # Get the email_message table
        email_message_table = getattr(db, "email_message", None)
        if email_message_table is None:
            # If no email_message table exists, return 0 unread count
            return {"unread_count": 0}

        # Count unread emails in the specified folder
        # Iterate through all emails and count those that:
        # 1. Belong to the specified folder (folder_id matches)
        # 2. Are not read (is_read == False)
        # 3. Are not deleted (is_deleted == False) - deleted emails shouldn't be counted
        unread_count = 0
        for message_id, email_message in email_message_table.items():
            # Check if email belongs to the target folder and is unread
            if (email_message.folder_id == folder_id and 
                not email_message.is_read and 
                not email_message.is_deleted):
                unread_count += 1

        # Return the unread count as specified in the schema
        return {"unread_count": unread_count}

    @is_tool()
    def retrieve_email_by_id(self, message_id: str):
        """
        Fetch complete email details including headers, body, and metadata by message ID.

        Args:
            message_id: Unique identifier of the email message

        Returns:
            dict: Email details containing message_id, subject, from_address, to_addresses,
                  body, received_at, and is_read status

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

        # Get the email_message table from the database
        email_table = getattr(db, "email_message", None)

        # Check if the email_message table exists
        if email_table is None:
            raise KeyError(f"Email message table not found in database")

        # Check if the message_id exists in the email_message table
        if message_id not in email_table:
            raise KeyError(f"Email with message_id '{message_id}' not found in mailbox")

        # Retrieve the email message object
        email = email_table[message_id]

        # Format the received_at timestamp to "yyyy-mm-dd HH:MM:SS" format
        # Handle case where received_at might be None
        received_at_str = None
        if email.received_at is not None:
            received_at_str = email.received_at.strftime("%Y-%m-%d %H:%M:%S")

        # Construct the return dictionary with all required fields
        # Note: to_addresses maps to to_recipients in the database schema
        result = {
            "message_id": email.message_id,
            "subject": email.subject,
            "from_address": email.from_address,
            "to_addresses": email.to_recipients,  # Map to_recipients to to_addresses
            "body": email.body,
            "received_at": received_at_str,
            "is_read": email.is_read
        }

        return result

    @is_tool()
    def rename_email_folder(self, folder_id: str, new_name: str):
        """
        Rename an existing email folder by updating its folder_name attribute.

        Args:
            folder_id: Identifier of the folder to rename
            new_name: New name for the folder

        Returns:
            dict: {"success": True} if rename operation succeeded

        Raises:
            KeyError: If folder_id does not exist in the database
            ValueError: If new_name is empty or invalid
        """
        # Access the database instance
        db = self.db

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

        # Strip whitespace from new_name
        new_name = new_name.strip()
        if not new_name:
            raise ValueError("new_name cannot be empty or contain only whitespace")

        # Get the email_folder table from database
        email_folder_table = getattr(db, "email_folder", None)

        # Check if email_folder table exists
        if email_folder_table is None:
            raise KeyError(f"Folder with folder_id '{folder_id}' does not exist")

        # Check if the folder exists in the table
        if folder_id not in email_folder_table:
            raise KeyError(f"Folder with folder_id '{folder_id}' does not exist")

        # Retrieve the folder object
        folder = email_folder_table[folder_id]

        # Check if this is a system folder (system folders typically cannot be renamed)
        if folder.is_system_folder:
            raise ValueError(f"Cannot rename system folder '{folder.folder_name}'")

        # Update the folder name
        folder.folder_name = new_name

        # Write the updated folder back to the database
        email_folder_table[folder_id] = folder
        setattr(db, "email_folder", email_folder_table)

        # Return success response
        return {"success": True}

    @is_tool()
    def list_calendar_events_by_date_range(self, start_date: str, end_date: str):
        """
        Get all calendar events within a specified date range.

        This method retrieves all calendar events that fall within the specified date range
        (inclusive). It filters events based on their start_time being within the range.

        Args:
            start_date: Start date of range in yyyy-mm-dd HH:MM:SS format
            end_date: End date of range in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Dictionary containing a list of event objects in the date range
                  Format: {"events": [{"event_id": "...", "title": "...", ...}, ...]}

        Raises:
            ValueError: If date format is invalid or start_date is after end_date
        """
        from datetime import datetime

        # Validate and parse start_date
        try:
            start_datetime = datetime.strptime(start_date, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid start_date format: {start_date}. Expected format: yyyy-mm-dd HH:MM:SS")

        # Validate and parse end_date
        try:
            end_datetime = datetime.strptime(end_date, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid end_date format: {end_date}. Expected format: yyyy-mm-dd HH:MM:SS")

        # Validate that start_date is not after end_date
        if start_datetime > end_datetime:
            raise ValueError(f"start_date ({start_date}) cannot be after end_date ({end_date})")

        # Access the database
        db = self.db

        # Get the calendar_event table
        calendar_event_table = getattr(db, "calendar_event", None)

        # If table doesn't exist or is empty, return empty list
        if calendar_event_table is None or not calendar_event_table:
            return {"events": []}

        # Filter events within the date range
        # An event is included if its start_time falls within [start_datetime, end_datetime]
        filtered_events = []

        for event_id, event in calendar_event_table.items():
            # Check if event's start_time is within the specified range
            if start_datetime <= event.start_time <= end_datetime:
                # Convert event object to dictionary format for return
                event_dict = {
                    "event_id": event.event_id,
                    "title": event.title,
                    "description": event.description,
                    "start_time": event.start_time.strftime("%Y-%m-%d %H:%M:%S"),
                    "end_time": event.end_time.strftime("%Y-%m-%d %H:%M:%S"),
                    "location": event.location,
                    "attendees": event.attendees,
                    "is_recurring": event.is_recurring,
                    "series_id": event.series_id,
                    "created_at": event.created_at.strftime("%Y-%m-%d %H:%M:%S")
                }
                filtered_events.append(event_dict)

        # Sort events by start_time for consistent ordering
        filtered_events.sort(key=lambda x: x["start_time"])

        # Return the list of events in the required format
        return {"events": filtered_events}

    @is_tool()
    def validate_email_address(self, email_address: str):
        """
        Validate the format and syntax of an email address.

        This method checks if the provided email address conforms to standard email format:
        - Contains exactly one '@' symbol
        - Has non-empty local part (before @) and domain part (after @)
        - Domain part contains at least one dot
        - Local and domain parts contain valid characters
        - No leading/trailing whitespace or dots in local/domain parts

        Args:
            email_address: Email address string to validate

        Returns:
            dict: Dictionary containing:
                - is_valid (bool): True if email is valid, False otherwise
                - error_message (str): Empty string if valid, error description if invalid

        Raises:
            ValueError: If email_address is None or not a string
        """
        import re

        # Input validation: ensure email_address is provided and is a string
        if email_address is None:
            raise ValueError("Email address cannot be None")

        if not isinstance(email_address, str):
            raise ValueError("Email address must be a string")

        # Strip leading/trailing whitespace for validation
        email_address = email_address.strip()

        # Check if email address is empty after stripping
        if not email_address:
            return {
                "is_valid": False,
                "error_message": "Email address cannot be empty"
            }

        # Check for exactly one '@' symbol
        if email_address.count('@') != 1:
            return {
                "is_valid": False,
                "error_message": "Email address must contain exactly one '@' symbol"
            }

        # Split email into local and domain parts
        local_part, domain_part = email_address.split('@')

        # Validate local part (before @)
        if not local_part:
            return {
                "is_valid": False,
                "error_message": "Local part (before '@') cannot be empty"
            }

        # Check local part doesn't start or end with dot
        if local_part.startswith('.') or local_part.endswith('.'):
            return {
                "is_valid": False,
                "error_message": "Local part cannot start or end with a dot"
            }

        # Check for consecutive dots in local part
        if '..' in local_part:
            return {
                "is_valid": False,
                "error_message": "Local part cannot contain consecutive dots"
            }

        # Validate local part characters (alphanumeric, dots, hyphens, underscores, plus signs)
        # According to RFC 5322, local part can contain: a-z A-Z 0-9 . _ % + -
        local_pattern = re.compile(r'^[a-zA-Z0-9._%-]+$')
        if not local_pattern.match(local_part):
            return {
                "is_valid": False,
                "error_message": "Local part contains invalid characters"
            }

        # Validate domain part (after @)
        if not domain_part:
            return {
                "is_valid": False,
                "error_message": "Domain part (after '@') cannot be empty"
            }

        # Check domain part contains at least one dot
        if '.' not in domain_part:
            return {
                "is_valid": False,
                "error_message": "Domain part must contain at least one dot"
            }

        # Check domain doesn't start or end with dot or hyphen
        if domain_part.startswith('.') or domain_part.endswith('.'):
            return {
                "is_valid": False,
                "error_message": "Domain part cannot start or end with a dot"
            }

        if domain_part.startswith('-') or domain_part.endswith('-'):
            return {
                "is_valid": False,
                "error_message": "Domain part cannot start or end with a hyphen"
            }

        # Check for consecutive dots in domain
        if '..' in domain_part:
            return {
                "is_valid": False,
                "error_message": "Domain part cannot contain consecutive dots"
            }

        # Validate domain part characters (alphanumeric, dots, hyphens)
        domain_pattern = re.compile(r'^[a-zA-Z0-9.-]+$')
        if not domain_pattern.match(domain_part):
            return {
                "is_valid": False,
                "error_message": "Domain part contains invalid characters"
            }

        # Split domain into labels (parts separated by dots)
        domain_labels = domain_part.split('.')

        # Validate each domain label
        for label in domain_labels:
            # Each label must be non-empty
            if not label:
                return {
                    "is_valid": False,
                    "error_message": "Domain labels cannot be empty"
                }

            # Each label cannot start or end with hyphen
            if label.startswith('-') or label.endswith('-'):
                return {
                    "is_valid": False,
                    "error_message": "Domain labels cannot start or end with hyphen"
                }

            # Each label must be 63 characters or less (RFC 1035)
            if len(label) > 63:
                return {
                    "is_valid": False,
                    "error_message": "Domain labels must be 63 characters or less"
                }

        # Check top-level domain (last label) is at least 2 characters and contains only letters
        tld = domain_labels[-1]
        if len(tld) < 2:
            return {
                "is_valid": False,
                "error_message": "Top-level domain must be at least 2 characters"
            }

        if not tld.isalpha():
            return {
                "is_valid": False,
                "error_message": "Top-level domain must contain only letters"
            }

        # Check overall email length (RFC 5321: maximum 254 characters)
        if len(email_address) > 254:
            return {
                "is_valid": False,
                "error_message": "Email address exceeds maximum length of 254 characters"
            }

        # Check local part length (maximum 64 characters per RFC 5321)
        if len(local_part) > 64:
            return {
                "is_valid": False,
                "error_message": "Local part exceeds maximum length of 64 characters"
            }

        # All validations passed
        return {
            "is_valid": True,
            "error_message": ""
        }

    @is_tool()
    def remove_attachment_from_draft(self, draft_id: str, attachment_id: str):
        """
        Remove an attachment from an email draft.

        This method removes a specified attachment from a draft email by:
        1. Verifying the draft exists
        2. Verifying the attachment exists and belongs to the draft
        3. Removing the attachment from the database

        Args:
            draft_id: Identifier of the draft
            attachment_id: Identifier of the attachment to remove

        Returns:
            dict: {"success": True} if removal succeeded

        Raises:
            KeyError: If draft or attachment does not exist, or attachment doesn't belong to draft
        """
        # Access the database
        db = self.db

        # Get the email_draft table
        email_draft_table = getattr(db, "email_draft", None)
        if email_draft_table is None:
            raise KeyError(f"email_draft table not found in database")

        # Get the email_attachment table
        email_attachment_table = getattr(db, "email_attachment", None)
        if email_attachment_table is None:
            raise KeyError(f"email_attachment table not found in database")

        # Verify the draft exists
        if draft_id not in email_draft_table:
            raise KeyError(f"Draft with draft_id '{draft_id}' does not exist")

        # Verify the attachment exists
        if attachment_id not in email_attachment_table:
            raise KeyError(f"Attachment with attachment_id '{attachment_id}' does not exist")

        # Get the attachment to verify it belongs to the specified draft
        attachment = email_attachment_table[attachment_id]

        # Check if the attachment belongs to the specified draft
        if attachment.draft_id != draft_id:
            raise KeyError(f"Attachment '{attachment_id}' does not belong to draft '{draft_id}'")

        # Remove the attachment from the database
        # Create a new dictionary without the attachment to be removed
        updated_attachments = {
            att_id: att 
            for att_id, att in email_attachment_table.items() 
            if att_id != attachment_id
        }

        # Update the email_attachment table
        setattr(db, "email_attachment", updated_attachments)

        # Return success response
        return {"success": True}

    @is_tool()
    def remove_flag_from_email(self, message_id: str):
        """
        Remove flag or star marker from an email

        This method removes the flag/star from a specified email by setting its
        is_flagged attribute to False. The email must exist and be currently flagged.

        Args:
            message_id: Unique identifier of the email message to unflag

        Returns:
            dict: Contains 'success' boolean indicating whether flag was removed

        Raises:
            KeyError: If the email with given message_id does not exist
        """
        # Access the database instance
        db = self.db

        # Retrieve the email_message table from database
        email_message_table = getattr(db, "email_message", None)

        # Check if the email_message table exists
        if email_message_table is None:
            raise KeyError(f"Email message table does not exist in database")

        # Check if the message exists in the table
        if message_id not in email_message_table:
            raise KeyError(f"Email message with ID '{message_id}' does not exist")

        # Retrieve the email message object
        email = email_message_table[message_id]

        # Remove the flag by setting is_flagged to False
        email.is_flagged = False

        # Update the email in the database
        email_message_table[message_id] = email
        setattr(db, "email_message", email_message_table)

        # Return success response
        return {
            "success": True
        }

    @is_tool()
    def delete_custom_label(self, label_id: str):
        """
        Delete a custom label from the system and remove it from all associated emails.

        Args:
            label_id: Identifier of the label to delete

        Returns:
            dict: {"success": bool} indicating whether the deletion succeeded

        Raises:
            KeyError: If label doesn't exist or is a system label
        """
        # Access the database
        db = self.db

        # Get the email_label table
        email_label_table = getattr(db, "email_label", None)
        if email_label_table is None:
            raise KeyError(f"Email label table not found in database")

        # Check if the label exists
        if label_id not in email_label_table:
            raise KeyError(f"Label with id '{label_id}' does not exist")

        # Get the label object
        label = email_label_table[label_id]

        # Check if it's a system label (system labels cannot be deleted)
        if label.is_system_label:
            raise KeyError(f"Cannot delete system label '{label_id}'")

        # Get the message_label table to remove all associations
        message_label_table = getattr(db, "message_label", None)
        if message_label_table is not None:
            # Find and collect all message_label entries that reference this label
            message_label_ids_to_delete = []
            for message_label_id, message_label in message_label_table.items():
                if message_label.label_id == label_id:
                    message_label_ids_to_delete.append(message_label_id)

            # Delete all message_label associations
            for message_label_id in message_label_ids_to_delete:
                del message_label_table[message_label_id]

            # Update the message_label table in database
            setattr(db, "message_label", message_label_table)

        # Delete the label from email_label table
        del email_label_table[label_id]

        # Update the email_label table in database
        setattr(db, "email_label", email_label_table)

        # Return success response
        return {"success": True}

    @is_tool()
    def compose_email_draft(
        self,
        subject: str,
        body: str,
        to_recipients: List[str],
        cc_recipients: Optional[List[str]] = None,
        bcc_recipients: Optional[List[str]] = None,
        priority: Optional[Literal["low", "normal", "high"]] = "normal"
    ) -> Dict[str, str]:
        """
        Create a new email draft with subject, body, recipients, and optional attachments.

        Args:
            subject: Subject line of the email
            body: Main content of the email message
            to_recipients: List of primary recipient email addresses
            cc_recipients: Optional list of CC recipient email addresses
            bcc_recipients: Optional list of BCC recipient email addresses
            priority: Priority level of the email (must be "low", "normal", or "high")

        Returns:
            Dictionary containing draft_id and created_at timestamp

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

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

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

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

        # Validate all recipient email addresses in to_recipients
        for email in to_recipients:
            if not email or not isinstance(email, str) or '@' not in email:
                raise ValueError(f"Invalid email address in to_recipients: {email}")

        # Validate optional cc_recipients if provided
        if cc_recipients is not None:
            if not isinstance(cc_recipients, list):
                raise ValueError("cc_recipients must be a list")
            for email in cc_recipients:
                if not email or not isinstance(email, str) or '@' not in email:
                    raise ValueError(f"Invalid email address in cc_recipients: {email}")

        # Validate optional bcc_recipients if provided
        if bcc_recipients is not None:
            if not isinstance(bcc_recipients, list):
                raise ValueError("bcc_recipients must be a list")
            for email in bcc_recipients:
                if not email or not isinstance(email, str) or '@' not in email:
                    raise ValueError(f"Invalid email address in bcc_recipients: {email}")

        # Validate priority parameter - ensure it matches the enum values
        valid_priorities = ["low", "normal", "high"]
        if priority not in valid_priorities:
            raise ValueError(f"Priority must be one of {valid_priorities}, got: {priority}")

        # Generate unique draft_id using secure random token
        draft_id = "draft_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Get current timestamp
        created_at = datetime.now()

        # Access the database
        db = self.db

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

        # Get the email_draft table
        email_draft_table = getattr(db, "email_draft")

        # Create new EmailDraft object
        new_draft = EmailDraft(
            draft_id=draft_id,
            subject=subject,
            body=body,
            to_recipients=to_recipients,
            cc_recipients=cc_recipients if cc_recipients else None,
            bcc_recipients=bcc_recipients if bcc_recipients else None,
            priority=priority,
            created_at=created_at,
            parent_message_id=None,
            is_reply=False,
            is_forward=False
        )

        # Store the draft in the database
        email_draft_table[draft_id] = new_draft
        setattr(db, "email_draft", email_draft_table)

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

    @is_tool()
    def add_attachment_to_draft(self, draft_id: str, file_path: str, file_name: str = None):
        """
        Attach a file to an existing email draft

        Args:
            draft_id: Identifier of the draft to attach file to
            file_path: Path to the file to attach
            file_name: Display name for the attachment (optional, defaults to file name from path)

        Returns:
            dict: Contains attachment_id of the newly created attachment

        Raises:
            OSError: If file cannot be accessed or read
            ValueError: If draft does not exist or parameters are invalid
        """
        import os
        import secrets
        import hashlib
        from datetime import datetime
        from pathlib import Path

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

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

        # Access the database
        db = self.db

        # Retrieve email_draft table
        email_draft_table = getattr(db, "email_draft", None)
        if email_draft_table is None:
            raise ValueError("email_draft table does not exist in database")

        # Check if draft exists
        draft = email_draft_table.get(draft_id)
        if draft is None:
            raise ValueError(f"Draft with id '{draft_id}' does not exist")

        # Validate file path and check if file exists and is accessible
        file_path_obj = Path(file_path)
        if not file_path_obj.exists():
            raise OSError(f"File does not exist: {file_path}")

        if not file_path_obj.is_file():
            raise OSError(f"Path is not a file: {file_path}")

        # Try to access the file to ensure it's readable
        try:
            with open(file_path, 'rb') as f:
                # Just check if we can open it
                pass
        except PermissionError:
            raise OSError(f"Permission denied when accessing file: {file_path}")
        except Exception as e:
            raise OSError(f"Cannot access file {file_path}: {str(e)}")

        # Get file size
        try:
            file_size = file_path_obj.stat().st_size
        except Exception as e:
            raise OSError(f"Cannot get file size for {file_path}: {str(e)}")

        # Determine file name for display
        if file_name is None or file_name.strip() == "":
            # Use the actual file name from the path
            display_name = file_path_obj.name
        else:
            display_name = file_name

        # Determine MIME type based on file extension
        mime_type_map = {
            '.pdf': 'application/pdf',
            '.doc': 'application/msword',
            '.docx': 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
            '.xls': 'application/vnd.ms-excel',
            '.xlsx': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
            '.ppt': 'application/vnd.ms-powerpoint',
            '.pptx': 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
            '.txt': 'text/plain',
            '.jpg': 'image/jpeg',
            '.jpeg': 'image/jpeg',
            '.png': 'image/png',
            '.gif': 'image/gif',
            '.zip': 'application/zip',
            '.csv': 'text/csv',
            '.json': 'application/json',
            '.xml': 'application/xml',
            '.html': 'text/html',
            '.mp3': 'audio/mpeg',
            '.mp4': 'video/mp4',
        }

        file_extension = file_path_obj.suffix.lower()
        mime_type = mime_type_map.get(file_extension, 'application/octet-stream')

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

        # Create new EmailAttachment object
        new_attachment = EmailAttachment(
            attachment_id=attachment_id,
            draft_id=draft_id,
            message_id=None,  # This is a draft attachment, not yet sent
            file_name=display_name,
            file_path=file_path,
            file_size=file_size,
            mime_type=mime_type,
            created_at=datetime.now()
        )

        # Retrieve email_attachment table
        email_attachment_table = getattr(db, "email_attachment", None)
        if email_attachment_table is None:
            # Initialize the table if it doesn't exist
            email_attachment_table = {}

        # Add the new attachment to the table
        email_attachment_table[attachment_id] = new_attachment

        # Update the database
        setattr(db, "email_attachment", email_attachment_table)

        # Return the attachment_id
        return {
            "attachment_id": attachment_id
        }

    @is_tool()
    def add_attendee_to_event(self, event_id: str, email_address: str):
        """
        Add a new attendee to an existing calendar event.

        Args:
            event_id: Identifier of the event to add attendee to
            email_address: Email address of the attendee to add

        Returns:
            dict: {"success": bool} indicating whether the attendee was added successfully

        Raises:
            KeyError: If the event_id does not exist in the database
        """
        import re

        # Get the database instance
        db = self.db

        # Retrieve the calendar_event table from database
        calendar_event_table = getattr(db, "calendar_event", None)

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

        # Check if the event exists in the table
        if event_id not in calendar_event_table:
            raise KeyError(f"Event with event_id '{event_id}' does not exist")

        # Retrieve the event object
        event = calendar_event_table[event_id]

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

        # Initialize attendees list if it doesn't exist or is None
        if event.attendees is None:
            event.attendees = []

        # Check if the attendee already exists in the list (case-insensitive comparison)
        # This prevents duplicate additions
        attendee_emails_lower = [email.lower() for email in event.attendees]
        if email_address.lower() in attendee_emails_lower:
            # Attendee already exists, but we still return success as the desired state is achieved
            return {"success": True}

        # Add the new attendee to the event's attendees list
        event.attendees.append(email_address)

        # Update the event in the database
        # Since we modified the event object directly, we need to update it in the table
        calendar_event_table[event_id] = event
        setattr(db, "calendar_event", calendar_event_table)

        # Return success response
        return {"success": True}

    @is_tool()
    def list_all_labels(self):
        """
        Retrieve list of all available email labels from the database.

        This method fetches all email labels stored in the email_label table
        and returns them as a list of dictionaries containing label_id and name.

        Returns:
            dict: A dictionary containing:
                - labels (list): List of label objects, each with:
                    - label_id (str): Unique identifier of the label
                    - name (str): Name of the label

        Raises:
            RuntimeError: If mailbox is not accessible or database operation fails
        """
        try:
            # Access the database instance
            db = self.db

            # Retrieve the email_label table from database
            email_label_table = getattr(db, "email_label", None)

            # Check if email_label table exists and is accessible
            if email_label_table is None:
                raise RuntimeError("Email label table is not accessible in the mailbox")

            # Initialize the result list
            labels = []

            # Iterate through all labels in the table
            # email_label_table is a Dict[str, EmailLabel] where key is label_id
            for label_id, label_obj in email_label_table.items():
                # Extract label information and create label dictionary
                label_dict = {
                    "label_id": label_obj.label_id,
                    "name": label_obj.label_name  # Note: schema uses label_name field
                }
                labels.append(label_dict)

            # Return the list of all labels
            return {"labels": labels}

        except AttributeError as e:
            # Handle case where database attributes are not accessible
            raise RuntimeError(f"Failed to access email label data: {str(e)}")
        except Exception as e:
            # Handle any other unexpected errors during label retrieval
            raise RuntimeError(f"Error retrieving email labels: {str(e)}")

    @is_tool()
    def create_email_signature(self, signature_name: str, signature_content: str, is_default: bool = False) -> dict:
        """
        Create a custom email signature template

        Args:
            signature_name: Name to identify the signature
            signature_content: HTML or plain text content of the signature
            is_default: Whether this should be the default signature (default: False)

        Returns:
            dict: Dictionary containing the signature_id of the created signature

        Raises:
            ValueError: If signature_name is empty or signature_content is empty
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

        # Access the database
        db = self.db

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

        # If this signature should be the default, update all existing signatures to not be default
        if is_default:
            for existing_sig_id, existing_sig in email_signature_table.items():
                if existing_sig.is_default:
                    # Create a new signature object with is_default set to False
                    updated_sig = EmailSignature(
                        signature_id=existing_sig.signature_id,
                        signature_name=existing_sig.signature_name,
                        signature_content=existing_sig.signature_content,
                        is_default=False,
                        created_at=existing_sig.created_at
                    )
                    email_signature_table[existing_sig_id] = updated_sig

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

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

        # Get current timestamp
        created_at = datetime.now()

        # Create the new EmailSignature object
        new_signature = EmailSignature(
            signature_id=signature_id,
            signature_name=signature_name.strip(),
            signature_content=signature_content.strip(),
            is_default=is_default,
            created_at=created_at
        )

        # Add the new signature to the table
        email_signature_table[signature_id] = new_signature

        # Save the updated table back to the database
        setattr(db, "email_signature", email_signature_table)

        # Return the signature_id
        return {
            "signature_id": signature_id
        }

    @is_tool()
    def mark_email_as_read(self, message_id: str):
        """
        Mark a specific email message as read by updating its is_read status to True.

        Args:
            message_id: Unique identifier of the email message to mark as read

        Returns:
            dict: Contains 'success' boolean indicating operation result

        Raises:
            KeyError: If the email message with given message_id does not exist
        """
        # Access the database instance
        db = self.db

        # Retrieve the email_message table from the database
        email_message_table = getattr(db, 'email_message', None)

        # Check if the email_message table exists
        if email_message_table is None:
            raise KeyError(f"Email message table does not exist in database")

        # Check if the message exists in the table
        if message_id not in email_message_table:
            raise KeyError(f"Email message with message_id '{message_id}' does not exist")

        # Retrieve the specific email message
        email = email_message_table[message_id]

        # Update the is_read status to True
        email.is_read = True

        # Update the email_message table with the modified email
        email_message_table[message_id] = email
        setattr(db, 'email_message', email_message_table)

        # Return success response
        return {"success": True}

    @is_tool()
    def remove_attendee_from_event(self, event_id: str, email_address: str):
        """
        Remove an attendee from a calendar event.

        This method removes a specific attendee (identified by email address) from an event's
        attendee list. If the event or attendee doesn't exist, appropriate errors are raised.

        Args:
            event_id: Identifier of the event to modify
            email_address: Email address of the attendee to remove

        Returns:
            dict: Contains 'success' boolean indicating whether removal was successful

        Raises:
            KeyError: If event doesn't exist or attendee is not in the event
        """
        # Access the database
        db = self.db

        # Retrieve the calendar_event table from database
        calendar_event_table = getattr(db, "calendar_event", None)

        # Check if the calendar_event table exists
        if calendar_event_table is None:
            raise KeyError(f"Calendar event table not found in database")

        # Check if the event exists in the table
        if event_id not in calendar_event_table:
            raise KeyError(f"Event with id '{event_id}' does not exist")

        # Retrieve the event object
        event = calendar_event_table[event_id]

        # Check if the event has an attendees list
        if event.attendees is None:
            raise KeyError(f"Event '{event_id}' has no attendees list")

        # Check if the attendee email exists in the event's attendee list
        if email_address not in event.attendees:
            raise KeyError(f"Attendee '{email_address}' is not in event '{event_id}'")

        # Remove the attendee from the list
        # Create a new list without the specified email address
        updated_attendees = [email for email in event.attendees if email != email_address]

        # Update the event's attendees list
        event.attendees = updated_attendees

        # Write the updated event back to the database
        calendar_event_table[event_id] = event
        setattr(db, "calendar_event", calendar_event_table)

        # Return success response
        return {
            "success": True
        }

    @is_tool()
    def parse_email_headers(self, raw_headers: str):
        """
        Parse and extract structured information from email headers

        This method takes raw email headers as a string and parses them into a structured
        dictionary format. It handles common email header fields like From, To, Cc, Bcc,
        Subject, Date, Reply-To, Message-ID, etc.

        Args:
            raw_headers: Raw email headers as string with newline-separated header fields

        Returns:
            Dictionary containing parsed_headers with extracted header field information

        Raises:
            ValueError: If raw_headers is empty or invalid format
        """
        # Import email parsing library
        from email.parser import Parser
        from email import policy

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

        raw_headers = raw_headers.strip()
        if not raw_headers:
            raise ValueError("raw_headers cannot be empty or contain only whitespace")

        try:
            # Use email.parser to parse the headers
            # The default policy handles various encoding and formatting issues
            parser = Parser(policy=policy.default)

            # Parse the headers - add empty body to make it valid email format
            msg = parser.parsestr(raw_headers + "\n\n")

            # Initialize the parsed headers dictionary
            parsed_headers = {}

            # Extract common header fields
            # From field
            if msg.get("From"):
                parsed_headers["from"] = msg.get("From")

            # To field
            if msg.get("To"):
                parsed_headers["to"] = msg.get("To")

            # Cc field
            if msg.get("Cc"):
                parsed_headers["cc"] = msg.get("Cc")

            # Bcc field
            if msg.get("Bcc"):
                parsed_headers["bcc"] = msg.get("Bcc")

            # Subject field
            if msg.get("Subject"):
                parsed_headers["subject"] = msg.get("Subject")

            # Date field
            if msg.get("Date"):
                parsed_headers["date"] = msg.get("Date")

            # Reply-To field
            if msg.get("Reply-To"):
                parsed_headers["reply_to"] = msg.get("Reply-To")

            # Message-ID field
            if msg.get("Message-ID"):
                parsed_headers["message_id"] = msg.get("Message-ID")

            # In-Reply-To field (for threaded conversations)
            if msg.get("In-Reply-To"):
                parsed_headers["in_reply_to"] = msg.get("In-Reply-To")

            # References field (for threaded conversations)
            if msg.get("References"):
                parsed_headers["references"] = msg.get("References")

            # Content-Type field
            if msg.get("Content-Type"):
                parsed_headers["content_type"] = msg.get("Content-Type")

            # Content-Transfer-Encoding field
            if msg.get("Content-Transfer-Encoding"):
                parsed_headers["content_transfer_encoding"] = msg.get("Content-Transfer-Encoding")

            # MIME-Version field
            if msg.get("MIME-Version"):
                parsed_headers["mime_version"] = msg.get("MIME-Version")

            # Priority/Importance field
            if msg.get("Priority"):
                parsed_headers["priority"] = msg.get("Priority")
            elif msg.get("X-Priority"):
                parsed_headers["priority"] = msg.get("X-Priority")
            elif msg.get("Importance"):
                parsed_headers["importance"] = msg.get("Importance")

            # Return-Path field
            if msg.get("Return-Path"):
                parsed_headers["return_path"] = msg.get("Return-Path")

            # Received field (can have multiple, store as list)
            received_headers = msg.get_all("Received")
            if received_headers:
                parsed_headers["received"] = received_headers

            # Extract any other custom headers (X-* headers)
            for key in msg.keys():
                if key.startswith("X-") and key not in parsed_headers:
                    parsed_headers[key.lower()] = msg.get(key)

            # If no headers were successfully parsed, raise an error
            if not parsed_headers:
                raise ValueError("No valid email headers found in the input")

            return {
                "parsed_headers": parsed_headers
            }

        except Exception as e:
            # If parsing fails, raise ValueError with descriptive message
            raise ValueError(f"Failed to parse email headers: {str(e)}")

    @is_tool()
    def delete_email_folder(self, folder_id: str, delete_contents: bool = False):
        """
        Delete an email folder and optionally its contents.

        This method removes a folder from the email system. System folders (like inbox, trash)
        cannot be deleted. If delete_contents is False and the folder contains emails,
        the deletion may fail depending on the email system's behavior.

        Args:
            folder_id: Identifier of the folder to delete
            delete_contents: Whether to delete all emails in the folder (default: False)

        Returns:
            dict: {"success": bool} indicating whether the deletion succeeded

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

        # Get the email_folder table from database
        email_folders = getattr(db, "email_folder", None)

        # Check if email_folder table exists
        if email_folders is None:
            raise KeyError(f"Email folder table does not exist in database")

        # Check if the folder exists
        if folder_id not in email_folders:
            raise KeyError(f"Folder with id '{folder_id}' does not exist")

        # Get the folder object
        folder = email_folders[folder_id]

        # Check if it's a system folder - system folders cannot be deleted
        if folder.is_system_folder:
            raise ValueError(f"Cannot delete system folder '{folder.folder_name}'")

        # Note: delete_contents parameter is acknowledged but actual email deletion
        # would require access to an email table which is not provided in the schema.
        # In a real implementation, if delete_contents is True, we would:
        # 1. Query all emails in this folder
        # 2. Delete each email
        # If delete_contents is False and emails exist, some systems might prevent deletion

        # Create a new dictionary without the deleted folder
        updated_folders = {fid: folder_obj for fid, folder_obj in email_folders.items() 
                          if fid != folder_id}

        # Update the database by setting the new folder dictionary
        setattr(db, "email_folder", updated_folders)

        # Return success result
        return {"success": True}

    @is_tool()
    def set_event_reminder(self, event_id: str, reminder_minutes: int, reminder_method: Literal["email", "popup", "notification"] = "popup") -> dict:
        """
        Set a reminder notification for a calendar event.

        This method creates a new reminder for an existing calendar event. It validates
        that the event exists, generates a unique reminder ID, and stores the reminder
        configuration in the database.

        Args:
            event_id: Identifier of the event to set reminder for
            reminder_minutes: Minutes before event to trigger reminder (must be positive)
            reminder_method: Method of reminder delivery (email/popup/notification)

        Returns:
            dict: Contains reminder_id of the newly created reminder

        Raises:
            KeyError: If event_id does not exist in calendar
            ValueError: If reminder_minutes is not positive or reminder_method is invalid
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access database instance
        db = self.db

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

        # Validate reminder_method is in allowed enum values (safety protection)
        allowed_methods = ["email", "popup", "notification"]
        if reminder_method not in allowed_methods:
            raise ValueError(f"reminder_method must be one of {allowed_methods}, got '{reminder_method}'")

        # Get calendar_event table from database
        calendar_event_table = getattr(db, "calendar_event", None)
        if calendar_event_table is None:
            raise KeyError(f"calendar_event table not found in database")

        # Check if event exists in calendar
        if event_id not in calendar_event_table:
            raise KeyError(f"Event with event_id '{event_id}' does not exist in calendar")

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

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

        # Create new EventReminder instance
        new_reminder = EventReminder(
            reminder_id=reminder_id,
            event_id=event_id,
            reminder_minutes=reminder_minutes,
            reminder_method=reminder_method,
            created_at=datetime.now()
        )

        # Add reminder to table
        event_reminder_table[reminder_id] = new_reminder

        # Save updated table back to database
        setattr(db, "event_reminder", event_reminder_table)

        # Return reminder_id in expected format
        return {
            "reminder_id": reminder_id
        }

    @is_tool()
    def archive_email(self, message_id: str):
        """
        Archive an email to remove it from inbox while keeping it searchable.

        This method marks an email as archived by setting its is_archived flag to True.
        The email will be removed from the inbox view but remains accessible through search.

        Args:
            message_id: Identifier of the email to archive

        Returns:
            dict: A dictionary containing:
                - success (bool): Whether the archiving operation succeeded

        Raises:
            KeyError: If the email with the given message_id does not exist
        """
        # Access the database instance
        db = self.db

        # Retrieve the email_message table from the database
        email_message_table = getattr(db, "email_message", None)

        # Check if the email_message table exists
        if email_message_table is None:
            raise KeyError(f"Email message table not found in database")

        # Check if the message exists in the table
        if message_id not in email_message_table:
            raise KeyError(f"Email with message_id '{message_id}' does not exist")

        # Retrieve the email message object
        email = email_message_table[message_id]

        # Mark the email as archived by setting is_archived to True
        email.is_archived = True

        # Update the email_message table with the modified email
        email_message_table[message_id] = email
        setattr(db, "email_message", email_message_table)

        # Return success status
        return {
            "success": True
        }

    @is_tool()
    def extract_email_addresses_from_text(self, text: str):
        """
        Extract all email addresses from a text string using regular expressions.

        This method scans through the provided text and identifies all valid email addresses
        based on standard email format patterns. It returns a list of unique email addresses
        found in the text.

        Args:
            text: Text string to extract email addresses from

        Returns:
            dict: Dictionary containing 'email_addresses' key with list of extracted email addresses

        Raises:
            ValueError: If text parameter is None or not a string
        """
        import re

        # Validate input parameter
        if text is None:
            raise ValueError("Text parameter cannot be None")

        if not isinstance(text, str):
            raise ValueError(f"Text parameter must be a string, got {type(text).__name__}")

        # Regular expression pattern for matching email addresses
        # This pattern matches standard email formats: username@domain.extension
        # - Username part: alphanumeric characters, dots, hyphens, underscores, plus signs
        # - @ symbol
        # - Domain part: alphanumeric characters, dots, hyphens
        # - Extension: at least 2 characters (e.g., .com, .org, .co.uk)
        email_pattern = r'[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}'

        # Find all matches in the text
        matches = re.findall(email_pattern, text)

        # Remove duplicates while preserving order using dict.fromkeys()
        # This ensures each email address appears only once in the result
        unique_emails = list(dict.fromkeys(matches))

        # Return the extracted email addresses in the required format
        return {
            "email_addresses": unique_emails
        }

    @is_tool()
    def update_email_signature(self, signature_id: str, signature_content: str):
        """
        Update an existing email signature with new content.

        Args:
            signature_id: Identifier of the signature to update
            signature_content: New content for the signature (HTML or plain text)

        Returns:
            dict: {"success": bool} indicating whether the update succeeded

        Raises:
            KeyError: If the signature with the given signature_id does not exist
        """
        # Access the database
        db = self.db

        # Retrieve the email_signature table from the database
        email_signature_table = getattr(db, "email_signature", None)

        # Check if the email_signature table exists
        if email_signature_table is None:
            raise KeyError("Email signature table does not exist in the database")

        # Check if the signature with the given signature_id exists
        if signature_id not in email_signature_table:
            raise KeyError(f"Signature with ID '{signature_id}' does not exist")

        # Retrieve the existing signature object
        existing_signature = email_signature_table[signature_id]

        # Update the signature content directly on the existing object
        existing_signature.signature_content = signature_content

        # Update the signature in the database
        email_signature_table[signature_id] = existing_signature
        setattr(db, "email_signature", email_signature_table)

        # Return success status
        return {"success": True}

    @is_tool()
    def reply_to_email(self, message_id: str, reply_body: str, reply_all: bool = False):
        """
        Create a reply draft to an existing email with quoted original content.

        Args:
            message_id: Identifier of the email being replied to
            reply_body: Content of the reply message
            reply_all: Whether to reply to all recipients or just sender (default: False)

        Returns:
            dict: Contains draft_id of the created reply draft

        Raises:
            KeyError: If the original message does not exist
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve email_message table
        email_message_table = getattr(db, "email_message", None)
        if email_message_table is None:
            raise KeyError(f"email_message table does not exist in database")

        # Check if the original message exists
        if message_id not in email_message_table:
            raise KeyError(f"Email message with ID '{message_id}' does not exist")

        # Get the original message
        original_message = email_message_table[message_id]

        # Prepare reply subject (add "Re: " prefix if not already present)
        reply_subject = original_message.subject
        if not reply_subject.startswith("Re: "):
            reply_subject = f"Re: {reply_subject}"

        # Prepare recipients based on reply_all flag
        if reply_all:
            # Reply to all: include original sender and all original recipients (excluding self)
            to_recipients = [original_message.from_address]

            # Add original TO recipients (excluding duplicates)
            for recipient in original_message.to_recipients:
                if recipient not in to_recipients:
                    to_recipients.append(recipient)

            # Prepare CC recipients from original CC list (excluding duplicates)
            cc_recipients = []
            if original_message.cc_recipients:
                for cc_recipient in original_message.cc_recipients:
                    if cc_recipient not in to_recipients and cc_recipient not in cc_recipients:
                        cc_recipients.append(cc_recipient)

            # Set cc_recipients to None if empty list
            cc_recipients = cc_recipients if cc_recipients else None
        else:
            # Reply only to sender
            to_recipients = [original_message.from_address]
            cc_recipients = None

        # Format the original message content as quoted text
        quoted_body = f"\n\n--- Original Message ---\n"
        quoted_body += f"From: {original_message.from_address}\n"
        quoted_body += f"Sent: {original_message.sent_at.strftime('%Y-%m-%d %H:%M:%S') if original_message.sent_at else 'N/A'}\n"
        quoted_body += f"To: {', '.join(original_message.to_recipients)}\n"
        if original_message.cc_recipients:
            quoted_body += f"CC: {', '.join(original_message.cc_recipients)}\n"
        quoted_body += f"Subject: {original_message.subject}\n\n"
        quoted_body += original_message.body

        # Combine reply body with quoted original content
        full_reply_body = reply_body + quoted_body

        # Generate unique draft ID
        draft_id = "draft_reply_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create the reply draft using EmailDraft class (already imported at file header)
        reply_draft = EmailDraft(
            draft_id=draft_id,
            subject=reply_subject,
            body=full_reply_body,
            to_recipients=to_recipients,
            cc_recipients=cc_recipients,
            bcc_recipients=None,
            priority=original_message.priority,  # Inherit priority from original message
            created_at=datetime.now(),
            parent_message_id=message_id,
            is_reply=True,
            is_forward=False
        )

        # Retrieve or initialize email_draft table
        email_draft_table = getattr(db, "email_draft", None)
        if email_draft_table is None:
            email_draft_table = {}

        # Add the new draft to the table
        email_draft_table[draft_id] = reply_draft

        # Save the updated table back to database
        setattr(db, "email_draft", email_draft_table)

        # Return the draft ID
        return {"draft_id": draft_id}

    @is_tool()
    def mark_email_as_unread(self, message_id: str):
        """
        Mark a specific email message as unread.

        This method updates the is_read flag of an email message to False,
        effectively marking it as unread in the email system.

        Args:
            message_id: Unique identifier of the email message to mark as unread

        Returns:
            dict: A dictionary containing:
                - success (bool): True if operation succeeded

        Raises:
            KeyError: If the email message with the given message_id does not exist
        """
        # Access the database instance
        db = self.db

        # Retrieve the email_message table from the database
        email_message_table = getattr(db, "email_message", None)

        # Check if the email_message table exists
        if email_message_table is None:
            raise KeyError(f"Email message table does not exist in database")

        # Check if the message exists in the table
        if message_id not in email_message_table:
            raise KeyError(f"Email message with ID '{message_id}' does not exist")

        # Retrieve the specific email message
        email = email_message_table[message_id]

        # Mark the email as unread by setting is_read to False
        email.is_read = False

        # Update the email message in the database
        email_message_table[message_id] = email
        setattr(db, "email_message", email_message_table)

        # Return success status
        return {"success": True}

    @is_tool()
    def list_email_folders(self, include_system_folders: bool = True) -> dict:
        """
        Retrieve list of all email folders in the mailbox.

        Args:
            include_system_folders: Whether to include system folders like inbox and trash

        Returns:
            dict: Dictionary containing list of folder objects with folder_id and name

        Raises:
            RuntimeError: If mailbox is not accessible or database operation fails
        """
        try:
            # Access the database
            db = self.db

            # Get the email_folder table from database
            email_folder_table = getattr(db, 'email_folder', None)

            # Check if email_folder table exists
            if email_folder_table is None:
                raise RuntimeError("Email folder table is not accessible in the mailbox")

            # Initialize the result list
            folders = []

            # Iterate through all folders in the email_folder table
            for folder_id, folder_item in email_folder_table.items():
                # Check if we should include this folder based on include_system_folders parameter
                # If include_system_folders is False, skip system folders
                if not include_system_folders and folder_item.is_system_folder:
                    continue

                # Create folder object with required fields
                folder_obj = {
                    "folder_id": folder_item.folder_id,
                    "name": folder_item.folder_name
                }

                # Add to result list
                folders.append(folder_obj)

            # Return the list of folders
            return {"folders": folders}

        except AttributeError as e:
            # Handle attribute access errors
            raise RuntimeError(f"Failed to access mailbox data: {str(e)}")
        except Exception as e:
            # Handle any other unexpected errors
            raise RuntimeError(f"An error occurred while retrieving email folders: {str(e)}")

    @is_tool()
    def delete_email_signature(self, signature_id: str):
        """
        Delete an email signature from the database.

        This method removes a signature entry from the email_signature table based on
        the provided signature_id. It verifies the signature exists before deletion
        and returns the success status.

        Args:
            signature_id: Unique identifier of the signature to delete

        Returns:
            dict: Contains 'success' boolean indicating deletion result

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

        # Retrieve the email_signature table from the database
        email_signature_table = getattr(db, "email_signature", None)

        # Check if the email_signature table exists
        if email_signature_table is None:
            raise KeyError(f"Email signature table does not exist in the database")

        # Check if the signature_id exists in the table
        if signature_id not in email_signature_table:
            raise KeyError(f"Signature with id '{signature_id}' does not exist")

        # Delete the signature from the table by removing it from the dictionary
        del email_signature_table[signature_id]

        # Update the database with the modified table
        setattr(db, "email_signature", email_signature_table)

        # Return success status
        return {
            "success": True
        }

    @is_tool()
    def check_calendar_availability(self, start_time: str, end_time: str):
        """
        Check if a time slot is available in the calendar by verifying if any existing
        calendar events overlap with the specified time range.

        Args:
            start_time: Start time to check in yyyy-mm-dd HH:MM:SS format
            end_time: End time to check in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Contains 'is_available' (bool) and 'conflicting_events' (list of event IDs)

        Raises:
            ValueError: If time format is invalid or end_time is before start_time
        """
        from datetime import datetime

        # Validate and parse time strings
        try:
            start_dt = datetime.strptime(start_time, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid start_time format: {start_time}. Expected format: yyyy-mm-dd HH:MM:SS")

        try:
            end_dt = datetime.strptime(end_time, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid end_time format: {end_time}. Expected format: yyyy-mm-dd HH:MM:SS")

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

        # Access the database
        db = self.db

        # Get calendar_event table from database
        calendar_events = getattr(db, "calendar_event", None)

        # Initialize result
        conflicting_events = []

        # If no calendar events exist, the time slot is available
        if calendar_events is None or len(calendar_events) == 0:
            return {
                "is_available": True,
                "conflicting_events": []
            }

        # Check each event for time conflicts
        # Two time ranges overlap if:
        # - The new event starts before the existing event ends, AND
        # - The new event ends after the existing event starts
        for event_id, event in calendar_events.items():
            # Check if there's an overlap between the requested time slot and this event
            if start_dt < event.end_time and end_dt > event.start_time:
                # There is a conflict - the time ranges overlap
                conflicting_events.append(event_id)

        # Determine availability based on whether any conflicts were found
        is_available = len(conflicting_events) == 0

        return {
            "is_available": is_available,
            "conflicting_events": conflicting_events
        }

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

        # Retrieve the email_message table from the database
        email_message_table = getattr(db, "email_message", None)
        if email_message_table is None:
            raise KeyError(f"email_message table not found in database")

        # Check if the email message exists
        if message_id not in email_message_table:
            raise KeyError(f"Email message with message_id '{message_id}' does not exist")

        # Retrieve the email_attachment table from the database
        email_attachment_table = getattr(db, "email_attachment", None)
        if email_attachment_table is None:
            raise KeyError(f"email_attachment table not found in database")

        # Initialize the list to store attachment metadata
        attachments = []

        # Iterate through all attachments in the email_attachment table
        for attachment_id, attachment in email_attachment_table.items():
            # Check if the attachment belongs to the specified email message
            if attachment.message_id == message_id:
                # Create attachment metadata dictionary
                attachment_metadata = {
                    "attachment_id": attachment.attachment_id,
                    "file_name": attachment.file_name,
                    "size": attachment.file_size
                }
                # Add the metadata to the list
                attachments.append(attachment_metadata)

        # Return the list of attachments
        # Note: The list may be empty if the email has no attachments
        return {
            "attachments": attachments
        }

    @is_tool()
    def format_email_date(self, date_string: str, format_type: Literal["rfc2822", "iso8601", "human_readable"]):
        """
        Format a date string into standard email date format.

        This method takes a date string in "yyyy-mm-dd HH:MM:SS" format and converts it
        to one of three standard email date formats: RFC 2822, ISO 8601, or human-readable format.

        Args:
            date_string: Date string in "yyyy-mm-dd HH:MM:SS" format (e.g., "2024-01-15 10:30:00")
            format_type: Desired output format type, must be one of:
                        - "rfc2822": RFC 2822 format (e.g., "Mon, 15 Jan 2024 10:30:00 +0000")
                        - "iso8601": ISO 8601 format (e.g., "2024-01-15T10:30:00+00:00")
                        - "human_readable": Human-friendly format (e.g., "January 15, 2024 at 10:30 AM")

        Returns:
            dict: Dictionary containing the formatted date string
                  {"formatted_date": "formatted date string"}

        Raises:
            ValueError: If date_string is invalid, empty, or format_type is not in the enum list
        """
        from datetime import datetime

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

        # Strip whitespace from date_string
        date_string = date_string.strip()

        # Validate format_type (safety protection for enum parameter)
        valid_formats = ["rfc2822", "iso8601", "human_readable"]
        if format_type not in valid_formats:
            raise ValueError(f"format_type must be one of {valid_formats}, got '{format_type}'")

        # Parse the input date string
        # Support both full datetime format and date-only format
        try:
            # Try parsing full datetime format first (yyyy-mm-dd HH:MM:SS)
            if len(date_string) > 10 and ' ' in date_string:
                dt = datetime.strptime(date_string, "%Y-%m-%d %H:%M:%S")
            else:
                # Try parsing date-only format (yyyy-mm-dd)
                dt = datetime.strptime(date_string, "%Y-%m-%d")
        except ValueError as e:
            raise ValueError(f"Invalid date string format. Expected 'yyyy-mm-dd HH:MM:SS' or 'yyyy-mm-dd', got '{date_string}'")

        # Format the date according to the requested format_type
        if format_type == "rfc2822":
            # RFC 2822 format: "Day, DD Mon YYYY HH:MM:SS +0000"
            # Example: "Mon, 15 Jan 2024 10:30:00 +0000"
            weekday = dt.strftime("%a")  # Abbreviated weekday name (Mon, Tue, etc.)
            day = dt.strftime("%d")      # Day of month (01-31)
            month = dt.strftime("%b")    # Abbreviated month name (Jan, Feb, etc.)
            year = dt.strftime("%Y")     # 4-digit year
            time = dt.strftime("%H:%M:%S")  # Time in HH:MM:SS format
            formatted_date = f"{weekday}, {day} {month} {year} {time} +0000"

        elif format_type == "iso8601":
            # ISO 8601 format: "YYYY-MM-DDTHH:MM:SS+00:00"
            # Example: "2024-01-15T10:30:00+00:00"
            formatted_date = dt.strftime("%Y-%m-%dT%H:%M:%S+00:00")

        elif format_type == "human_readable":
            # Human-readable format: "Month DD, YYYY at HH:MM AM/PM"
            # Example: "January 15, 2024 at 10:30 AM"
            month_name = dt.strftime("%B")  # Full month name (January, February, etc.)
            day = dt.strftime("%d").lstrip("0")  # Day without leading zero
            year = dt.strftime("%Y")  # 4-digit year
            time_12hr = dt.strftime("%I:%M").lstrip("0")  # 12-hour format without leading zero
            am_pm = dt.strftime("%p")  # AM or PM
            formatted_date = f"{month_name} {day}, {year} at {time_12hr} {am_pm}"

        # Return the formatted date in the required dictionary format
        return {
            "formatted_date": formatted_date
        }

    @is_tool()
    def forward_email(self, message_id: str, forward_to: List[str], forward_message: str = None) -> Dict[str, str]:
        """
        Create a forward draft of an existing email to new recipients.

        This method:
        1. Retrieves the original email message from the database
        2. Creates a new draft with forward formatting
        3. Includes the original email content with forward markers
        4. Optionally adds a custom forward message
        5. Returns the newly created draft ID

        Args:
            message_id: Identifier of the email being forwarded
            forward_to: List of recipient email addresses for forwarding
            forward_message: Additional message to include with the forward (optional)

        Returns:
            Dictionary containing the draft_id of the created forward draft

        Raises:
            KeyError: If the original message does not exist in the database
        """
        import secrets
        import hashlib
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the email_message table
        email_messages = getattr(db, "email_message", None)
        if email_messages is None:
            raise KeyError(f"Email message table not found in database")

        # Check if the original message exists
        if message_id not in email_messages:
            raise KeyError(f"Email message with ID '{message_id}' not found")

        # Get the original message
        original_message = email_messages[message_id]

        # Generate a unique draft ID using hash of random bytes
        draft_prefix = "draft_fwd_"
        draft_id = draft_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Construct the forward subject line
        # Add "Fwd: " prefix if not already present
        forward_subject = original_message.subject
        if not forward_subject.startswith("Fwd:"):
            forward_subject = f"Fwd: {forward_subject}"

        # Construct the forward body
        # Include optional forward message at the top
        forward_body_parts = []

        if forward_message:
            forward_body_parts.append(forward_message)
            forward_body_parts.append("\n\n")

        # Add forward header with original email metadata
        forward_body_parts.append("---------- Forwarded message ---------\n")
        forward_body_parts.append(f"From: {original_message.from_address}\n")

        # Format sent_at datetime if available
        sent_at_str = original_message.sent_at.strftime('%Y-%m-%d %H:%M:%S') if original_message.sent_at else 'N/A'
        forward_body_parts.append(f"Date: {sent_at_str}\n")
        forward_body_parts.append(f"Subject: {original_message.subject}\n")
        forward_body_parts.append(f"To: {', '.join(original_message.to_recipients)}\n")

        # Add CC recipients if present
        if original_message.cc_recipients:
            forward_body_parts.append(f"Cc: {', '.join(original_message.cc_recipients)}\n")

        forward_body_parts.append("\n")

        # Add original message body
        forward_body_parts.append(original_message.body)

        forward_body = "".join(forward_body_parts)

        # Create the forward draft with EmailDraft class
        new_draft = EmailDraft(
            draft_id=draft_id,
            subject=forward_subject,
            body=forward_body,
            to_recipients=forward_to,
            cc_recipients=None,  # No CC recipients by default for forwards
            bcc_recipients=None,  # No BCC recipients by default for forwards
            priority=original_message.priority,  # Inherit priority from original message
            created_at=datetime.now(),
            parent_message_id=message_id,  # Reference to the original message
            is_reply=False,
            is_forward=True  # Mark this draft as a forward
        )

        # Retrieve the email_draft table
        email_drafts = getattr(db, "email_draft", None)
        if email_drafts is None:
            # Initialize the table if it doesn't exist
            email_drafts = {}

        # Add the new draft to the database
        email_drafts[draft_id] = new_draft
        setattr(db, "email_draft", email_drafts)

        # Return the draft ID
        return {"draft_id": draft_id}

    @is_tool()
    def send_email(self, draft_id: str):
        """
        Send an email immediately from a draft.

        This method:
        1. Retrieves the draft from email_draft table
        2. Creates a new sent message in email_message table
        3. Removes the draft from email_draft table

        Args:
            draft_id: Identifier of the draft to send

        Returns:
            dict: Contains message_id and sent_at timestamp

        Raises:
            RuntimeError: If draft not found or sending fails
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access database
        db = self.db

        # Retrieve email_draft table
        email_draft_table = getattr(db, "email_draft", None)
        if email_draft_table is None:
            raise RuntimeError("Email draft table not found in database")

        # Find the draft by draft_id
        draft = email_draft_table.get(draft_id)
        if draft is None:
            raise RuntimeError(f"Draft with ID '{draft_id}' not found")

        # Generate unique message_id for the sent email
        message_id_prefix = "msg_"
        message_id = message_id_prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

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

        # Retrieve email_message table
        email_message_table = getattr(db, "email_message", None)
        if email_message_table is None:
            raise RuntimeError("Email message table not found in database")

        # Determine the from_address from the first recipient in the draft
        # In a real system, this would come from the authenticated user's email
        # For this implementation, we'll use a default value
        from_address = "user@example.com"

        # Create new EmailMessage object from draft data
        new_message = EmailMessage(
            message_id=message_id,
            subject=draft.subject,
            body=draft.body,
            from_address=from_address,
            to_recipients=draft.to_recipients,
            cc_recipients=draft.cc_recipients,
            bcc_recipients=draft.bcc_recipients,
            sent_at=sent_at,
            received_at=None,
            is_read=True,
            is_flagged=False,
            is_archived=False,
            is_spam=False,
            is_deleted=False,
            folder_id="sent",
            priority=draft.priority if draft.priority else "normal"
        )

        # Add the new message to email_message table
        email_message_table[message_id] = new_message
        setattr(db, "email_message", email_message_table)

        # Remove the draft from email_draft table
        del email_draft_table[draft_id]
        setattr(db, "email_draft", email_draft_table)

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

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

    @is_tool()
    def search_emails_by_keyword(self, keyword: str, search_fields: Optional[List[str]] = None, limit: Optional[int] = None):
        """
        Search emails by keyword in subject, body, or sender fields.
        Uses fuzzy matching for natural language text fields to improve robustness.

        Args:
            keyword: Search keyword or phrase to look for in emails
            search_fields: List of fields to search in (subject, body, sender, recipients)
                          If None, searches in all fields by default
            limit: Maximum number of results to return. If None, returns all matches

        Returns:
            dict: Contains 'results' (list of matching message IDs) and 'total_count' (total matches)

        Raises:
            ValueError: If keyword is empty or search_fields contains invalid field names
        """
        from thefuzz import fuzz

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

        keyword = keyword.strip().lower()

        # Validate and set default search_fields
        valid_fields = ["subject", "body", "sender", "recipients"]
        if search_fields is None:
            # Default: search in all fields
            search_fields = valid_fields
        else:
            # Validate that all provided fields are valid
            invalid_fields = [field for field in search_fields if field not in valid_fields]
            if invalid_fields:
                raise ValueError(f"Invalid search fields: {invalid_fields}. Valid fields are: {valid_fields}")

        # Access the email_message table from database
        db = self.db
        email_message_table = getattr(db, "email_message", None)

        if email_message_table is None or not email_message_table:
            # No emails in database, return empty results
            return {
                "results": [],
                "total_count": 0
            }

        # List to store matching message IDs with their match scores
        matches = []

        # Fuzzy matching threshold (can be adjusted based on requirements)
        FUZZY_THRESHOLD = 60  # Match score >= 60 is considered a match

        # Iterate through all emails and search for keyword
        for message_id, email in email_message_table.items():
            # Skip deleted or spam emails (they shouldn't appear in search results)
            if email.is_deleted or email.is_spam:
                continue

            max_score = 0  # Track the highest match score for this email

            # Search in subject field
            if "subject" in search_fields and email.subject:
                subject_lower = email.subject.lower()
                # Use fuzzy matching for subject (natural language text)
                score = fuzz.partial_ratio(keyword, subject_lower)
                max_score = max(max_score, score)

            # Search in body field
            if "body" in search_fields and email.body:
                body_lower = email.body.lower()
                # Use fuzzy matching for body (natural language text)
                score = fuzz.partial_ratio(keyword, body_lower)
                max_score = max(max_score, score)

            # Search in sender field (from_address)
            if "sender" in search_fields and email.from_address:
                sender_lower = email.from_address.lower()
                # For email addresses, use partial_ratio to match parts of the address
                score = fuzz.partial_ratio(keyword, sender_lower)
                max_score = max(max_score, score)

            # Search in recipients fields (to, cc, bcc)
            if "recipients" in search_fields:
                # Check to_recipients
                if email.to_recipients:
                    for recipient in email.to_recipients:
                        recipient_lower = recipient.lower()
                        score = fuzz.partial_ratio(keyword, recipient_lower)
                        max_score = max(max_score, score)

                # Check cc_recipients
                if email.cc_recipients:
                    for recipient in email.cc_recipients:
                        recipient_lower = recipient.lower()
                        score = fuzz.partial_ratio(keyword, recipient_lower)
                        max_score = max(max_score, score)

                # Check bcc_recipients
                if email.bcc_recipients:
                    for recipient in email.bcc_recipients:
                        recipient_lower = recipient.lower()
                        score = fuzz.partial_ratio(keyword, recipient_lower)
                        max_score = max(max_score, score)

            # If match score exceeds threshold, add to results
            if max_score >= FUZZY_THRESHOLD:
                matches.append((message_id, max_score))

        # Sort matches by score in descending order (highest match first)
        matches.sort(key=lambda x: x[1], reverse=True)

        # Extract message IDs from sorted matches
        all_matching_ids = [message_id for message_id, _ in matches]
        total_count = len(all_matching_ids)

        # Apply limit if specified
        if limit is not None and limit > 0:
            result_ids = all_matching_ids[:limit]
        else:
            result_ids = all_matching_ids

        return {
            "results": result_ids,
            "total_count": total_count
        }

    @is_tool()
    def create_email_filter_rule(
        self,
        rule_name: str,
        condition_field: Literal["subject", "sender", "body", "recipient"],
        condition_operator: Literal["contains", "equals", "starts_with", "ends_with"],
        condition_value: str,
        action_type: Literal["move_to_folder", "mark_as_read", "add_label", "delete"],
        action_parameter: str = None
    ):
        """
        Create an automatic filter rule to organize incoming emails.

        This method creates a new email filter rule with specified conditions and actions.
        The rule will be automatically activated upon creation.
        """
        import secrets
        import hashlib
        from datetime import datetime

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

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

        # Validate that action_parameter is provided for actions that require it
        if action_type == "move_to_folder" and not action_parameter:
            raise ValueError("action_parameter is required for move_to_folder action")

        if action_type == "add_label" and not action_parameter:
            raise ValueError("action_parameter is required for add_label action")

        # Validate enum values (safety protection)
        valid_condition_fields = ["subject", "sender", "body", "recipient"]
        if condition_field not in valid_condition_fields:
            raise ValueError(f"condition_field must be one of {valid_condition_fields}, got '{condition_field}'")

        valid_condition_operators = ["contains", "equals", "starts_with", "ends_with"]
        if condition_operator not in valid_condition_operators:
            raise ValueError(f"condition_operator must be one of {valid_condition_operators}, got '{condition_operator}'")

        valid_action_types = ["move_to_folder", "mark_as_read", "add_label", "delete"]
        if action_type not in valid_action_types:
            raise ValueError(f"action_type must be one of {valid_action_types}, got '{action_type}'")

        # Access database
        db = self.db

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

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

        # Ensure rule_id is unique
        while rule_id in email_filter_rule_table:
            rule_id = prefix + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

        # Create new filter rule instance
        new_rule = EmailFilterRule(
            rule_id=rule_id,
            rule_name=rule_name.strip(),
            condition_field=condition_field,
            condition_operator=condition_operator,
            condition_value=condition_value.strip(),
            action_type=action_type,
            action_parameter=action_parameter.strip() if action_parameter else None,
            is_active=True,
            created_at=datetime.now()
        )

        # Add new rule to the table
        email_filter_rule_table[rule_id] = new_rule

        # Update database
        setattr(db, "email_filter_rule", email_filter_rule_table)

        # Return the created rule_id
        return {
            "rule_id": rule_id
        }

    @is_tool()
    def create_calendar_event(
        self,
        title: str,
        start_time: str,
        end_time: str,
        location: Optional[str] = None,
        attendees: Optional[List[str]] = None,
        description: Optional[str] = None
    ) -> Dict[str, str]:
        """
        Create a new calendar event with the provided details.

        Args:
            title: Title of the calendar event
            start_time: Start time in "yyyy-mm-dd HH:MM:SS" format
            end_time: End time in "yyyy-mm-dd HH:MM:SS" format
            location: Optional location of the event
            attendees: Optional list of attendee email addresses
            description: Optional detailed description of the event

        Returns:
            Dictionary containing the event_id of the created event

        Raises:
            ValueError: If time format is invalid or end_time is before start_time
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Validate and parse start_time
        try:
            start_dt = datetime.strptime(start_time, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid start_time format: {start_time}. Expected format: yyyy-mm-dd HH:MM:SS")

        # Validate and parse end_time
        try:
            end_dt = datetime.strptime(end_time, "%Y-%m-%d %H:%M:%S")
        except ValueError:
            raise ValueError(f"Invalid end_time format: {end_time}. Expected format: yyyy-mm-dd HH:MM:SS")

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

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

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

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

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

        # Access the database
        db = self.db

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

        # Get the calendar_event table
        calendar_event_table = getattr(db, "calendar_event")

        # Create new CalendarEvent instance
        new_event = CalendarEvent(
            event_id=event_id,
            title=title.strip(),
            description=description.strip() if description else None,
            start_time=start_dt,
            end_time=end_dt,
            location=location.strip() if location else None,
            attendees=attendees if attendees else None,
            is_recurring=False,  # Default value for new events
            series_id=None,  # Not part of a recurring series by default
            created_at=created_at
        )

        # Add the new event to the calendar_event table
        calendar_event_table[event_id] = new_event

        # Update the database with the modified table
        setattr(db, "calendar_event", calendar_event_table)

        # Return the event_id
        return {"event_id": event_id}

    @is_tool()
    def unmark_email_as_spam(self, message_id: str):
        """
        Remove spam marking from an email and move it back to inbox.

        This method performs the following operations:
        1. Validates that the message exists in the database
        2. Verifies that the email is currently marked as spam
        3. Removes the spam marking (sets is_spam to False)
        4. Moves the email to the inbox folder
        5. Updates the email record in the database

        Args:
            message_id: Unique identifier of the email to unmark from spam

        Returns:
            dict: Contains 'success' boolean indicating operation result

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

        # Retrieve the email_message table from the database
        email_message_table = getattr(db, "email_message", None)

        # Validate that the email_message table exists
        if email_message_table is None:
            raise KeyError(f"Email message table not found in database")

        # Check if the message exists in the database
        if message_id not in email_message_table:
            raise KeyError(f"Email message with ID '{message_id}' not found")

        # Retrieve the email message object
        email = email_message_table[message_id]

        # Verify pre-condition: email must be currently marked as spam
        if not email.is_spam:
            # If email is not marked as spam, still return success
            # as the desired state (not spam) is already achieved
            return {"success": True}

        # Remove spam marking by setting is_spam to False
        email.is_spam = False

        # Move email back to inbox by updating folder_id
        # Assuming "inbox" is the standard inbox folder identifier
        email.folder_id = "inbox"

        # Update the email record in the database
        email_message_table[message_id] = email
        setattr(db, "email_message", email_message_table)

        # Return success result
        return {"success": True}

    @is_tool()
    def delete_email(self, message_id: str, permanent: bool = False):
        """
        Move an email to trash folder or permanently delete it

        Args:
            message_id: Unique identifier of the email message
            permanent: Whether to permanently delete (True) or move to trash (False)

        Returns:
            dict: Contains 'success' boolean indicating if deletion succeeded

        Raises:
            KeyError: If the email message with given message_id does not exist
        """
        # Get database instance
        db = self.db

        # Retrieve email_message table from database
        email_message_table = getattr(db, "email_message", None)

        # Check if email_message table exists
        if email_message_table is None:
            raise KeyError(f"Email message table not found in database")

        # Check if the message exists in the table
        if message_id not in email_message_table:
            raise KeyError(f"Email message with message_id '{message_id}' does not exist")

        # Get the email message object
        email = email_message_table[message_id]

        # Perform deletion based on permanent flag
        if permanent:
            # Permanently delete: remove the message from the table
            del email_message_table[message_id]
            # Update the database with modified table
            setattr(db, "email_message", email_message_table)
        else:
            # Move to trash: set is_deleted flag to True
            # Note: The email remains in the table but marked as deleted
            email.is_deleted = True
            # Update the email in the table
            email_message_table[message_id] = email
            # Update the database with modified table
            setattr(db, "email_message", email_message_table)

        # Return success result
        return {"success": True}

    @is_tool()
    def delete_email_filter_rule(self, rule_id: str):
        """
        Delete an existing email filter rule from the database.

        This method removes a filter rule identified by rule_id from the email_filter_rule table.
        If the rule does not exist, a KeyError is raised.

        Args:
            rule_id: Identifier of the rule to delete

        Returns:
            dict: A dictionary containing:
                - success (bool): Whether the deletion succeeded

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

        # Retrieve the email_filter_rule table from the database
        email_filter_rule_table = getattr(db, "email_filter_rule", None)

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

        # Check if the rule_id exists in the table
        if rule_id not in email_filter_rule_table:
            raise KeyError(f"Filter rule with rule_id '{rule_id}' does not exist")

        # Delete the rule from the table by removing it from the dictionary
        del email_filter_rule_table[rule_id]

        # Update the database with the modified table
        setattr(db, "email_filter_rule", email_filter_rule_table)

        # Return success response
        return {
            "success": True
        }

    @is_tool()
    def set_out_of_office_reply(self, message: str, start_date: str, end_date: str, reply_to_internal_only: bool = False):
        """
        Configure automatic out-of-office reply message.
        """
        from datetime import datetime
        import secrets
        import hashlib

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

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

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

        # Validate that start_date is not after end_date
        if start_dt > end_dt:
            raise ValueError(f"Start date ({start_date}) cannot be after end date ({end_date})")

        # Access the database
        db = self.db

        # Get the out_of_office_config table
        out_of_office_table = getattr(db, "out_of_office_config", None)

        # Initialize the table if it doesn't exist
        if out_of_office_table is None:
            out_of_office_table = {}

        # Deactivate all existing configurations
        # Only one out-of-office configuration should be active at a time
        for existing_config in out_of_office_table.values():
            existing_config.is_active = False

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

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

        # Create new out-of-office configuration
        # OutOfOfficeConfig class is already imported at file header
        new_config = OutOfOfficeConfig(
            config_id=config_id,
            message=message,
            start_date=start_dt,
            end_date=end_dt,
            reply_to_internal_only=reply_to_internal_only,
            is_active=True,  # New configuration is active
            created_at=created_at
        )

        # Add the new configuration to the table
        out_of_office_table[config_id] = new_config

        # Save the updated table back to the database
        setattr(db, "out_of_office_config", out_of_office_table)

        # Return success status
        return {"success": True}

    @is_tool()
    def create_recurring_event(
        self,
        title: str,
        start_time: str,
        end_time: str,
        recurrence_pattern: Literal["daily", "weekly", "monthly", "yearly"],
        recurrence_interval: int = 1,
        recurrence_end_date: str = None
    ):
        from datetime import datetime, timedelta
        from dateutil.relativedelta import relativedelta
        import secrets
        import hashlib

        # Validate recurrence_pattern first
        valid_patterns = ["daily", "weekly", "monthly", "yearly"]
        if recurrence_pattern not in valid_patterns:
            raise ValueError(f"Invalid recurrence_pattern. Must be one of: {', '.join(valid_patterns)}")

        # Validate and parse input parameters
        try:
            # Parse start and end times
            start_dt = datetime.strptime(start_time, "%Y-%m-%d %H:%M:%S")
            end_dt = datetime.strptime(end_time, "%Y-%m-%d %H:%M:%S")

            # Validate that end time is after start time
            if end_dt <= start_dt:
                raise ValueError("End time must be after start time")

            # Parse recurrence end date if provided
            end_date = None
            if recurrence_end_date:
                end_date = datetime.strptime(recurrence_end_date, "%Y-%m-%d").date()
                # Validate that recurrence end date is after start date
                if end_date < start_dt.date():
                    raise ValueError("Recurrence end date must be after or equal to start date")

            # Validate recurrence interval
            if recurrence_interval < 1:
                raise ValueError("Recurrence interval must be at least 1")

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

        except ValueError as e:
            # Re-raise ValueError with more context if it's a parsing error
            if "time data" in str(e) or "does not match format" in str(e):
                raise ValueError(f"Invalid date/time format: {str(e)}")
            raise

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

        # Access database
        db = self.db

        # Get existing tables or initialize empty dicts
        recurring_series_table = getattr(db, "recurring_event_series", None)
        if recurring_series_table is None:
            recurring_series_table = {}

        calendar_events_table = getattr(db, "calendar_event", None)
        if calendar_events_table is None:
            calendar_events_table = {}

        # Create recurring event series record
        series_record = RecurringEventSeries(
            series_id=series_id,
            title=title,
            recurrence_pattern=recurrence_pattern,
            recurrence_interval=recurrence_interval,
            recurrence_end_date=end_date,
            created_at=datetime.now()
        )

        # Save series to database
        recurring_series_table[series_id] = series_record
        setattr(db, "recurring_event_series", recurring_series_table)

        # Generate individual calendar events based on recurrence pattern
        # Calculate event duration
        event_duration = end_dt - start_dt

        # Initialize current occurrence time
        current_start = start_dt
        occurrence_count = 0
        max_occurrences = 1000  # Safety limit to prevent infinite loops

        # Generate events until end date or max occurrences
        while occurrence_count < max_occurrences:
            # Check if we've reached the end date
            if end_date and current_start.date() > end_date:
                break

            # Calculate end time for current occurrence
            current_end = current_start + event_duration

            # Generate unique event_id for this occurrence
            event_id = "event_" + hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]

            # Create calendar event for this occurrence
            event_record = CalendarEvent(
                event_id=event_id,
                title=title,
                start_time=current_start,
                end_time=current_end,
                is_recurring=True,
                series_id=series_id,
                created_at=datetime.now()
            )

            # Save event to database
            calendar_events_table[event_id] = event_record

            # Calculate next occurrence based on recurrence pattern
            if recurrence_pattern == "daily":
                current_start = current_start + timedelta(days=recurrence_interval)
            elif recurrence_pattern == "weekly":
                current_start = current_start + timedelta(weeks=recurrence_interval)
            elif recurrence_pattern == "monthly":
                current_start = current_start + relativedelta(months=recurrence_interval)
            elif recurrence_pattern == "yearly":
                current_start = current_start + relativedelta(years=recurrence_interval)

            occurrence_count += 1

        # Save updated calendar events table to database
        setattr(db, "calendar_event", calendar_events_table)

        # Return series_id
        return {"series_id": series_id}

    @is_tool()
    def filter_emails_by_date_range(self, start_date: str, end_date: str, folder_id: str = None):
        """
        Filter emails within a specific date range.

        This method searches for emails whose received_at timestamp falls within
        the specified date range. If folder_id is provided, only emails in that
        folder are considered.

        Args:
            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
            folder_id: Optional specific folder to search in

        Returns:
            dict: Contains 'message_ids' - list of email message IDs in the date range

        Raises:
            ValueError: If date format is invalid or start_date is after end_date
        """
        from datetime import datetime

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

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

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

        # Access the database
        db = self.db
        email_message_table = getattr(db, "email_message", None)

        # Handle case where email_message table doesn't exist or is empty
        if email_message_table is None or not email_message_table:
            return {"message_ids": []}

        # Filter emails based on date range and optional folder_id
        filtered_message_ids = []

        for message_id, email in email_message_table.items():
            # Skip if email doesn't have received_at timestamp
            if email.received_at is None:
                continue

            # Check if email is within date range
            if start_dt <= email.received_at <= end_dt:
                # If folder_id is specified, check if email is in that folder
                if folder_id is not None:
                    if email.folder_id == folder_id:
                        filtered_message_ids.append(message_id)
                else:
                    # No folder filter, include all emails in date range
                    filtered_message_ids.append(message_id)

        # Return the filtered message IDs
        return {"message_ids": filtered_message_ids}

    @is_tool()
    def update_calendar_event(self, event_id: str, title: str = None, start_time: str = None, end_time: str = None):
        """
        Update details of an existing calendar event

        Args:
            event_id: Identifier of the event to update
            title: New title for the event (optional)
            start_time: New start time in yyyy-mm-dd HH:MM:SS format (optional)
            end_time: New end time in yyyy-mm-dd HH:MM:SS format (optional)

        Returns:
            dict: {"success": bool} indicating whether the update succeeded

        Raises:
            KeyError: If the event_id does not exist in the calendar
            ValueError: If time format is invalid or start_time is not before end_time
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Get the calendar_event table
        calendar_event_table = getattr(db, "calendar_event", None)

        # Check if calendar_event table exists
        if calendar_event_table is None:
            raise KeyError(f"Calendar event table not found")

        # Check if the event exists
        if event_id not in calendar_event_table:
            raise KeyError(f"Event with id '{event_id}' does not exist")

        # Get the existing event
        event = calendar_event_table[event_id]

        # Update title if provided
        if title is not None:
            event.title = title

        # Parse and validate time updates
        parsed_start_time = None
        parsed_end_time = None

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

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

        # Determine final start and end times for validation
        final_start_time = parsed_start_time if parsed_start_time is not None else event.start_time
        final_end_time = parsed_end_time if parsed_end_time is not None else event.end_time

        # Validate that start_time is before end_time
        if final_start_time >= final_end_time:
            raise ValueError(f"Start time must be before end time")

        # Apply the validated updates
        if parsed_start_time is not None:
            event.start_time = parsed_start_time
        if parsed_end_time is not None:
            event.end_time = parsed_end_time

        # Update the event in the database
        calendar_event_table[event_id] = event
        setattr(db, "calendar_event", calendar_event_table)

        # Return success response
        return {"success": True}

    @is_tool()
    def add_label_to_email(self, message_id: str, label_name: str):
        """
        Add a custom label or tag to an email for categorization.

        Args:
            message_id: Identifier of the email to add the label to
            label_name: Name of the label to add to the email

        Returns:
            dict: {"success": bool} indicating whether the label was added successfully

        Raises:
            KeyError: If the email message or label does not exist
        """
        import secrets
        import hashlib
        from datetime import datetime
        from thefuzz import fuzz, process

        # Access the database
        db = self.db

        # Retrieve email_message table
        email_message_table = getattr(db, "email_message", None)
        if email_message_table is None:
            raise KeyError(f"Email message table does not exist")

        # Retrieve email_label table
        email_label_table = getattr(db, "email_label", None)
        if email_label_table is None:
            raise KeyError(f"Email label table does not exist")

        # Retrieve message_label table
        message_label_table = getattr(db, "message_label", None)
        if message_label_table is None:
            # Initialize empty message_label table if it doesn't exist
            message_label_table = {}
            setattr(db, "message_label", message_label_table)

        # Verify the email message exists (exact match on message_id)
        if message_id not in email_message_table:
            raise KeyError(f"Email message with id '{message_id}' does not exist")

        # Find the label by name using fuzzy matching for natural language text
        # Extract all label names and their corresponding label_ids
        label_candidates = {label.label_name: label.label_id 
                           for label in email_label_table.values()}

        if not label_candidates:
            raise KeyError(f"No labels exist in the system")

        # Use fuzzy matching to find the best matching label name
        best_match = process.extractOne(label_name, label_candidates.keys())

        # If no reasonable match found (score too low), raise KeyError
        if best_match is None or best_match[1] < 60:  # 60% similarity threshold
            raise KeyError(f"Label with name '{label_name}' does not exist")

        matched_label_name, similarity_score = best_match
        label_id = label_candidates[matched_label_name]

        # Check if this label is already added to the message
        # Iterate through message_label table to see if relationship already exists
        for ml in message_label_table.values():
            if ml.message_id == message_id and ml.label_id == label_id:
                # Label already exists on this message, return success
                return {"success": True}

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

        # Create new MessageLabel relationship
        new_message_label = MessageLabel(
            message_label_id=message_label_id,
            message_id=message_id,
            label_id=label_id,
            created_at=datetime.now()
        )

        # Add the new message_label to the table
        message_label_table[message_label_id] = new_message_label
        setattr(db, "message_label", message_label_table)

        # Return success
        return {"success": True}

    @is_tool()
    def list_email_filter_rules(self):
        """
        Retrieve all configured email filter rules from the database.

        This method fetches all email filter rules stored in the email_filter_rule table
        and returns them as a list of dictionaries containing rule information.

        Returns:
            dict: A dictionary containing:
                - rules: List of filter rule objects with all rule details

        Raises:
            RuntimeError: If mailbox is not accessible or database operation fails
        """
        try:
            # Access the database instance
            db = self.db

            # Retrieve the email_filter_rule table from the database
            email_filter_rule_table = getattr(db, "email_filter_rule", None)

            # Check if the table exists and is accessible
            if email_filter_rule_table is None:
                raise RuntimeError("Mailbox is not accessible: email_filter_rule table not found")

            # Initialize the list to store all filter rules
            rules_list = []

            # Iterate through all filter rules in the table
            # The table is a Dict[str, EmailFilterRule] where key is rule_id
            for rule_id, rule_obj in email_filter_rule_table.items():
                # Convert each EmailFilterRule object to a dictionary
                # Include all relevant fields from the schema
                rule_dict = {
                    "rule_id": rule_obj.rule_id,
                    "rule_name": rule_obj.rule_name,
                    "condition_field": rule_obj.condition_field,
                    "condition_operator": rule_obj.condition_operator,
                    "condition_value": rule_obj.condition_value,
                    "action_type": rule_obj.action_type,
                    "action_parameter": rule_obj.action_parameter,
                    "is_active": rule_obj.is_active,
                    "created_at": rule_obj.created_at.strftime("%Y-%m-%d %H:%M:%S")
                }

                # Add the rule dictionary to the list
                rules_list.append(rule_dict)

            # Return the list of rules wrapped in the expected format
            return {"rules": rules_list}

        except AttributeError as e:
            # Handle cases where database attributes are not accessible
            raise RuntimeError(f"Mailbox is not accessible: {str(e)}")
        except Exception as e:
            # Handle any other unexpected errors during database operation
            raise RuntimeError(f"Failed to retrieve email filter rules: {str(e)}")

    @is_tool()
    def disable_out_of_office_reply(self):
        """
        Disable the automatic out-of-office reply by setting is_active to False
        for all active out-of-office configurations.

        This method:
        1. Retrieves all out-of-office configurations from the database
        2. Finds any active configurations (is_active=True)
        3. Deactivates them by setting is_active=False
        4. Updates the database with the modified configurations

        Returns:
            dict: A dictionary containing:
                - success (bool): True if at least one active configuration was disabled,
                                False if no active configurations were found

        Raises:
            RuntimeError: If there's an error accessing or updating the database
        """
        try:
            # Access the database
            db = self.db

            # Get the out_of_office_config table
            out_of_office_table = getattr(db, "out_of_office_config", None)

            # Check if the table exists
            if out_of_office_table is None:
                raise RuntimeError("out_of_office_config table not found in database")

            # Check if table is empty
            if not out_of_office_table:
                return {"success": False}

            # Flag to track if any configuration was disabled
            disabled_any = False

            # Iterate through all configurations
            for config_id, config in out_of_office_table.items():
                # Check if this configuration is currently active
                if config.is_active:
                    # Disable the configuration by setting is_active to False
                    config.is_active = False
                    disabled_any = True

            # Update the database with modified configurations
            if disabled_any:
                setattr(db, "out_of_office_config", out_of_office_table)

            # Return success status
            return {"success": disabled_any}

        except AttributeError as e:
            # Handle attribute access errors
            raise RuntimeError(f"Error accessing database attributes: {str(e)}")
        except Exception as e:
            # Handle any other unexpected errors
            raise RuntimeError(f"Error disabling out-of-office reply: {str(e)}")

    @is_tool()
    def filter_emails_by_sender(self, email_address: str, folder_id: Optional[str] = None) -> dict:
        """
        Filter emails from a specific sender address.

        This method searches through the email_message database to find all emails
        from the specified sender email address. Optionally filters by folder.

        Args:
            email_address: Email address of the sender to filter by
            folder_id: Optional specific folder to search in (e.g., "inbox")

        Returns:
            dict: Dictionary containing list of message IDs from the sender
                  Format: {"message_ids": ["msg_111", "msg_222", ...]}

        Raises:
            ValueError: If email_address is empty or invalid format
        """
        # Validate input parameters
        if not email_address or not isinstance(email_address, str):
            raise ValueError("email_address must be a non-empty string")

        # Basic email format validation (contains @ symbol)
        if "@" not in email_address:
            raise ValueError("email_address must be a valid email format containing '@'")

        # Access the database
        db = self.db

        # Get the email_message table from database
        email_message_table = getattr(db, "email_message", None)

        # Initialize result list
        matching_message_ids = []

        # If email_message table exists and contains data
        if email_message_table:
            # Iterate through all email messages in the database
            for message_id, email_msg in email_message_table.items():
                # Check if the email is from the specified sender
                # Use exact match for email addresses (precision required for email identifiers)
                if email_msg.from_address == email_address:
                    # If folder_id is specified, filter by folder as well
                    if folder_id is not None:
                        # Only include emails in the specified folder
                        if email_msg.folder_id == folder_id:
                            matching_message_ids.append(message_id)
                    else:
                        # No folder filter, include all emails from sender
                        matching_message_ids.append(message_id)

        # Return the result in the required format
        return {
            "message_ids": matching_message_ids
        }

    @is_tool()
    def delete_calendar_event(self, event_id: str, notify_attendees: bool = False) -> dict:
        """
        Delete a calendar event from the database.

        Args:
            event_id: Identifier of the event to delete
            notify_attendees: Whether to send cancellation notice to attendees (default: False)

        Returns:
            dict: {"success": bool} - Whether the deletion succeeded

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

        # Get the calendar_event table from database
        calendar_events = getattr(db, 'calendar_event', None)

        # Check if calendar_event table exists
        if calendar_events is None:
            raise KeyError(f"Calendar event table does not exist in database")

        # Check if the event exists in the calendar
        if event_id not in calendar_events:
            raise KeyError(f"Event with id '{event_id}' does not exist in calendar")

        # Get the event before deletion (for potential notification purposes)
        event_to_delete = calendar_events[event_id]

        # If notify_attendees is True and event has attendees, we would send notifications
        # Note: Actual notification sending is not implemented as it's outside the scope
        # of this database operation, but we acknowledge the flag
        if notify_attendees and event_to_delete.attendees:
            # In a real implementation, this would trigger email notifications
            # to all attendees in event_to_delete.attendees list
            # For now, we just log this intent through a comment
            pass

        # Create a new dictionary without the deleted event
        updated_calendar_events = {
            key: value for key, value in calendar_events.items() 
            if key != event_id
        }

        # Update the database with the new calendar_events dictionary
        setattr(db, 'calendar_event', updated_calendar_events)

        # Return success status
        return {"success": True}

    @is_tool()
    def cancel_scheduled_email(self, scheduled_id: str):
        """
        Cancel a previously scheduled email send.

        This method cancels a scheduled email by marking it as cancelled in the database.
        It verifies that the scheduled send exists and has not been sent yet before cancellation.

        Args:
            scheduled_id: Identifier of the scheduled send to cancel

        Returns:
            dict: A dictionary containing:
                - success (bool): Whether the cancellation succeeded

        Raises:
            KeyError: If the scheduled_id does not exist in the database
            ValueError: If the scheduled email has already been sent
        """
        # Get the database instance
        db = self.db

        # Retrieve the scheduled_email table from the database
        scheduled_email_table = getattr(db, "scheduled_email", None)

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

        # Check if the scheduled_id exists in the table
        if scheduled_id not in scheduled_email_table:
            raise KeyError(f"Scheduled email with id '{scheduled_id}' does not exist")

        # Retrieve the scheduled email item
        scheduled_email_item = scheduled_email_table[scheduled_id]

        # Check if the email has already been sent
        # Cannot cancel an email that has already been sent
        if scheduled_email_item.is_sent:
            raise ValueError(f"Cannot cancel scheduled email '{scheduled_id}' because it has already been sent")

        # Mark the scheduled email as cancelled
        scheduled_email_item.is_cancelled = True

        # Update the scheduled email in the database
        scheduled_email_table[scheduled_id] = scheduled_email_item
        setattr(db, "scheduled_email", scheduled_email_table)

        # Return success status
        return {
            "success": True
        }

    @is_tool()
    def mark_email_as_spam(self, message_id: str):
        """
        Mark an email as spam and move it to spam folder.

        This method performs the following operations:
        1. Validates that the email exists in the database
        2. Marks the email as spam by setting is_spam to True
        3. Moves the email to the spam folder by updating folder_id
        4. Updates the email record in the database

        Args:
            message_id: Identifier of the email to mark as spam

        Returns:
            dict: A dictionary containing:
                - success (bool): Whether the operation succeeded

        Raises:
            KeyError: If the email with the given message_id does not exist
        """
        # Access the database instance
        db = self.db

        # Retrieve the email_message table from the database
        email_message_table = getattr(db, "email_message", None)

        # Check if the email_message table exists
        if email_message_table is None:
            raise KeyError(f"Email message table does not exist in database")

        # Check if the email with the given message_id exists in the table
        if message_id not in email_message_table:
            raise KeyError(f"Email with message_id '{message_id}' does not exist")

        # Retrieve the email message object
        email = email_message_table[message_id]

        # Mark the email as spam by setting is_spam flag to True
        email.is_spam = True

        # Move the email to spam folder by updating folder_id
        # Standard spam folder identifier is "spam"
        email.folder_id = "spam"

        # Update the email record in the database
        # The email object is already modified in place, but we ensure it's saved
        email_message_table[message_id] = email
        setattr(db, "email_message", email_message_table)

        # Return success status
        return {"success": True}

    @is_tool()
    def remove_label_from_email(self, message_id: str, label_name: str):
        """
        Remove a label or tag from an email by deleting the corresponding message_label relationship.

        Args:
            message_id: Identifier of the email
            label_name: Name of the label to remove

        Returns:
            dict: Contains 'success' boolean indicating whether the label was removed successfully

        Raises:
            KeyError: If the email message doesn't exist, label doesn't exist, or label is not associated with the message
        """
        # Get database instance
        db = self.db

        # Retrieve email_message table
        email_message_table = getattr(db, "email_message", None)
        if email_message_table is None or message_id not in email_message_table:
            raise KeyError(f"Email message with id '{message_id}' does not exist")

        # Retrieve email_label table
        email_label_table = getattr(db, "email_label", None)
        if email_label_table is None or len(email_label_table) == 0:
            raise KeyError(f"No labels exist in the system")

        # Find the label by name using exact matching
        # label_name is a user-defined tag name, should use exact match for precision
        target_label_id = None
        for label in email_label_table.values():
            if label.label_name == label_name:
                target_label_id = label.label_id
                break

        if target_label_id is None:
            raise KeyError(f"Label '{label_name}' does not exist")

        # Retrieve message_label table
        message_label_table = getattr(db, "message_label", None)
        if message_label_table is None or len(message_label_table) == 0:
            raise KeyError(f"Label '{label_name}' is not associated with email '{message_id}'")

        # Find the message_label relationship to remove
        message_label_id_to_remove = None
        for ml_id, message_label in message_label_table.items():
            if message_label.message_id == message_id and message_label.label_id == target_label_id:
                message_label_id_to_remove = ml_id
                break

        # If no relationship found, the label is not associated with this message
        if message_label_id_to_remove is None:
            raise KeyError(f"Label '{label_name}' is not associated with email '{message_id}'")

        # Remove the message_label relationship
        del message_label_table[message_label_id_to_remove]

        # Update the database with the modified message_label table
        setattr(db, "message_label", message_label_table)

        # Return success response
        return {
            "success": True
        }

    @is_tool()
    def get_mailbox_storage_quota(self):
        """
        Retrieve current mailbox storage usage and quota information.

        This method calculates the total storage used by email messages, drafts, and attachments,
        and returns storage statistics including used storage, total quota, and usage percentage.

        Returns:
            dict: A dictionary containing:
                - used_storage_mb: Current storage used in megabytes
                - total_quota_mb: Total storage quota in megabytes (15360 MB = 15 GB)
                - usage_percentage: Percentage of quota used

        Raises:
            RuntimeError: If there is an error accessing mailbox data or calculating storage
        """
        try:
            # Access the database
            db = self.db

            # Initialize total storage used in bytes
            total_storage_bytes = 0

            # Calculate storage from email messages
            # Each message contributes its subject, body, and metadata to storage
            email_messages = getattr(db, "email_message", None)
            if email_messages:
                for message_id, message in email_messages.items():
                    # Calculate approximate storage for message
                    # Subject + body + from_address + to/cc/bcc recipients
                    message_size = 0
                    message_size += len(message.subject.encode('utf-8'))
                    message_size += len(message.body.encode('utf-8'))
                    message_size += len(message.from_address.encode('utf-8'))

                    # Add recipient addresses
                    for recipient in message.to_recipients:
                        message_size += len(recipient.encode('utf-8'))
                    if message.cc_recipients:
                        for recipient in message.cc_recipients:
                            message_size += len(recipient.encode('utf-8'))
                    if message.bcc_recipients:
                        for recipient in message.bcc_recipients:
                            message_size += len(recipient.encode('utf-8'))

                    # Add metadata overhead (approximate 1KB per message)
                    message_size += 1024

                    total_storage_bytes += message_size

            # Calculate storage from email drafts
            # Drafts also contribute to storage usage
            email_drafts = getattr(db, "email_draft", None)
            if email_drafts:
                for draft_id, draft in email_drafts.items():
                    # Calculate approximate storage for draft
                    draft_size = 0
                    draft_size += len(draft.subject.encode('utf-8'))
                    draft_size += len(draft.body.encode('utf-8'))

                    # Add recipient addresses
                    for recipient in draft.to_recipients:
                        draft_size += len(recipient.encode('utf-8'))
                    if draft.cc_recipients:
                        for recipient in draft.cc_recipients:
                            draft_size += len(recipient.encode('utf-8'))
                    if draft.bcc_recipients:
                        for recipient in draft.bcc_recipients:
                            draft_size += len(recipient.encode('utf-8'))

                    # Add metadata overhead (approximate 1KB per draft)
                    draft_size += 1024

                    total_storage_bytes += draft_size

            # Calculate storage from attachments
            # Attachments are the main contributor to storage usage
            email_attachments = getattr(db, "email_attachment", None)
            if email_attachments:
                for attachment_id, attachment in email_attachments.items():
                    # Use the actual file size stored in the attachment
                    total_storage_bytes += attachment.file_size

            # Convert bytes to megabytes (1 MB = 1024 * 1024 bytes)
            used_storage_mb = total_storage_bytes / (1024 * 1024)

            # Set total quota to 15 GB (15360 MB)
            # This is a typical mailbox quota for email services
            total_quota_mb = 15360.0

            # Calculate usage percentage
            # Ensure we don't divide by zero and handle edge cases
            if total_quota_mb > 0:
                usage_percentage = (used_storage_mb / total_quota_mb) * 100
                # Round to one decimal place for cleaner output
                usage_percentage = round(usage_percentage, 1)
            else:
                usage_percentage = 0.0

            # Round used storage to one decimal place
            used_storage_mb = round(used_storage_mb, 1)

            # Return the storage quota information
            return {
                "used_storage_mb": used_storage_mb,
                "total_quota_mb": total_quota_mb,
                "usage_percentage": usage_percentage
            }

        except Exception as e:
            # Raise RuntimeError if there's any issue accessing or calculating storage
            raise RuntimeError(f"Failed to retrieve mailbox storage quota: {str(e)}")

    @is_tool()
    def schedule_email_send(self, draft_id: str, scheduled_time: str):
        """
        Schedule an email to be sent at a specific future time.

        This method creates a scheduled send entry for an existing draft email.
        It validates that the draft exists and the scheduled time is in the future.

        Args:
            draft_id: Identifier of the draft to schedule
            scheduled_time: Time to send the email in yyyy-mm-dd HH:MM:SS format

        Returns:
            dict: Contains scheduled_id of the created scheduled send

        Raises:
            ValueError: If draft doesn't exist or scheduled time is not in the future
        """
        from datetime import datetime
        import secrets
        import hashlib

        # Access the database
        db = self.db

        # Get the email_draft table
        email_draft_table = getattr(db, "email_draft", None)
        if email_draft_table is None:
            raise ValueError("Email draft table not found in database")

        # Validate that the draft exists
        if draft_id not in email_draft_table:
            raise ValueError(f"Draft with id '{draft_id}' does not exist")

        # Parse the scheduled time string to datetime object
        try:
            scheduled_datetime = datetime.strptime(scheduled_time, "%Y-%m-%d %H:%M:%S")
        except ValueError as e:
            raise ValueError(f"Invalid scheduled_time format. Expected 'yyyy-mm-dd HH:MM:SS', got '{scheduled_time}': {str(e)}")

        # Validate that scheduled time is in the future
        current_time = datetime.now()
        if scheduled_datetime <= current_time:
            raise ValueError(f"Scheduled time '{scheduled_time}' must be in the future. Current time is '{current_time.strftime('%Y-%m-%d %H:%M:%S')}'")

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

        # Create the ScheduledEmail entry
        scheduled_email = ScheduledEmail(
            scheduled_id=scheduled_id,
            draft_id=draft_id,
            scheduled_time=scheduled_datetime,
            is_sent=False,
            is_cancelled=False,
            created_at=current_time
        )

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

        # Add the new scheduled email to the table
        scheduled_email_table[scheduled_id] = scheduled_email

        # Update the database
        setattr(db, "scheduled_email", scheduled_email_table)

        # Return the scheduled_id
        return {
            "scheduled_id": scheduled_id
        }
