from typing import List, Dict, Optional, Any
from scale_env.environment.toolkit import ToolKitBase, ToolType, is_tool
from .database import *
from thefuzz import fuzz, process

"""Tools for smart_home."""

class SmartHomeTools(ToolKitBase):
    """All tools for smart_home."""
    
    db: SmartHomeDB
    
    def __init__(self, db: SmartHomeDB):
        """Initialize tools with database."""
        super().__init__(db)
    
    @is_tool()
    def set_appliance_eco_mode(self, appliance_id: str, eco_mode_enabled: bool) -> dict:
        """
        Enable or disable eco-friendly operation mode for energy efficiency.

        This method updates the eco_mode_enabled setting for a specific appliance
        in the database. It validates the appliance exists and updates its eco mode status.

        Args:
            appliance_id: Unique identifier of the appliance
            eco_mode_enabled: Whether to enable eco mode (True) or disable it (False)

        Returns:
            dict: Contains success status and current eco mode status
                - success (bool): Whether eco mode was set successfully
                - eco_mode_enabled (str): Current eco mode status ('enabled' or 'disabled')

        Raises:
            RuntimeError: If appliance is not found or database operation fails
        """
        # Input validation: ensure appliance_id is provided and not empty
        if not appliance_id or not isinstance(appliance_id, str):
            raise RuntimeError("Invalid appliance_id: must be a non-empty string")

        # Validate eco_mode_enabled is boolean
        if not isinstance(eco_mode_enabled, bool):
            raise RuntimeError("Invalid eco_mode_enabled: must be a boolean value")

        # Access the database
        db = self.db

        # Retrieve the appliance table from database
        appliance_table = getattr(db, 'appliance', None)

        # Check if appliance table exists
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if the appliance exists in the database
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' not found")

        # Retrieve the appliance object
        appliance = appliance_table[appliance_id]

        # Update the eco_mode_enabled field
        appliance.eco_mode_enabled = eco_mode_enabled

        # Write the updated appliance data back to the database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Determine the eco mode status string for return value
        eco_mode_status = 'enabled' if eco_mode_enabled else 'disabled'

        # Return success response with current eco mode status
        return {
            'success': True,
            'eco_mode_enabled': eco_mode_status
        }

    @is_tool()
    def get_appliance_energy_consumption(self, appliance_id: str) -> dict:
        """
        Retrieve the current energy consumption rate of an appliance.

        This method fetches the most recent energy consumption reading for a specific appliance
        from the appliance_energy_consumption table.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing:
                - current_power_watts: Current power consumption in watts
                - timestamp: Time of the reading in yyyy-mm-dd HH:MM:SS format

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

        # Retrieve the appliance_energy_consumption table
        energy_consumption_table = getattr(db, 'appliance_energy_consumption', None)

        # Check if the table exists and is not empty
        if energy_consumption_table is None:
            raise KeyError(f"Appliance energy consumption table not found")

        # Check if the appliance_id exists in the table
        if appliance_id not in energy_consumption_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found in energy consumption records")

        # Retrieve the energy consumption record for the specified appliance
        consumption_record = energy_consumption_table[appliance_id]

        # Extract the current power consumption in watts
        # Convert Decimal to float for consistent JSON serialization
        current_power_watts = float(consumption_record.current_power_watts)

        # Extract the timestamp and format it to yyyy-mm-dd HH:MM:SS format
        # The timestamp is already a datetime object in the database
        timestamp_str = consumption_record.timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Construct and return the result dictionary
        result = {
            'current_power_watts': current_power_watts,
            'timestamp': timestamp_str
        }

        return result

    @is_tool()
    def unlock_appliance(self, appliance_id: str) -> dict:
        """
        Unlock an appliance to enable physical button operations.

        This method changes the lock_status of the specified appliance from 'locked' to 'unlocked',
        allowing physical controls on the device to be used.

        Args:
            appliance_id: Unique identifier of the appliance to unlock

        Returns:
            dict: Contains success status and current lock status
                - success (bool): Whether the appliance was unlocked successfully
                - lock_status (str): Current lock status after operation

        Raises:
            RuntimeError: If appliance not found, not connected, or not in locked state
        """
        # Get database instance
        db = self.db

        # Retrieve appliance table from database
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if appliance exists
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Verify appliance is connected to network
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        # Check pre-condition: appliance must be in locked state
        if appliance.lock_status != 'locked':
            raise RuntimeError(
                f"Appliance '{appliance_id}' is not in locked state. "
                f"Current lock status: {appliance.lock_status}"
            )

        # Unlock the appliance by updating lock_status to 'unlocked'
        appliance.lock_status = 'unlocked'

        # Update the appliance in database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Return success result with current lock status
        return {
            'success': True,
            'lock_status': 'unlocked'
        }

    @is_tool()
    def get_appliance_humidity_level(self, appliance_id: str) -> dict:
        """
        Retrieve the current humidity level and target setting from an appliance.

        This method fetches humidity-related data for a specific appliance that supports
        humidity monitoring. It retrieves both the target humidity setting and the current
        humidity reading from the appliance database.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: A dictionary containing:
                - target_humidity (int): Target humidity setting percentage (0-100)
                - current_humidity (int): Current humidity reading percentage (0-100)

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

        # Retrieve the appliance table from the database
        appliance_table = getattr(db, 'appliance', None)

        # Check if the appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if the appliance_id exists in the appliance table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with id '{appliance_id}' not found")

        # Retrieve the appliance object
        appliance = appliance_table[appliance_id]

        # Extract humidity data from the appliance
        # Note: These fields may be None if the appliance doesn't support humidity monitoring
        target_humidity = appliance.target_humidity
        current_humidity = appliance.current_humidity

        # Construct and return the result dictionary
        # Return the values as-is (including None if not supported)
        result = {
            'target_humidity': target_humidity,
            'current_humidity': current_humidity
        }

        return result

    @is_tool()
    def get_appliance_power_state(self, appliance_id: str) -> dict:
        """
        Retrieve the current power state of a specific appliance.

        This method queries the appliance database to get the power state information
        for the specified appliance. It validates that the appliance exists and is
        connected to the network before returning the power state details.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing:
                - power_state: Current power state of the appliance (on, off, standby)
                - last_updated: Last time the power state was updated in yyyy-mm-dd HH:MM:SS format

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

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

        # Access the database
        db = self.db

        # Get the appliance table from database
        appliance_table = getattr(db, 'appliance', None)

        # Check if appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Retrieve the specific appliance by ID
        # The appliance table is a Dict[str, Appliance] where key is appliance_id
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Extract power state information
        power_state = appliance.power_state
        power_state_updated_at = appliance.power_state_updated_at

        # Convert datetime object to string format yyyy-mm-dd HH:MM:SS
        # Handle both datetime objects and potential string inputs
        if isinstance(power_state_updated_at, datetime):
            last_updated_str = power_state_updated_at.strftime("%Y-%m-%d %H:%M:%S")
        else:
            # If already a string, ensure it's in correct format
            last_updated_str = str(power_state_updated_at)

        # Construct and return the result dictionary
        result = {
            "power_state": power_state,
            "last_updated": last_updated_str
        }

        return result

    @is_tool()
    def get_appliance_lock_status(self, appliance_id: str) -> dict:
        """
        Retrieve the current lock status of an appliance.

        This method queries the appliance database to get the lock status of a specific appliance.
        It validates that the appliance exists and returns its current lock status.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing the lock status
                - lock_status: Current lock status (e.g., 'locked', 'unlocked')

        Raises:
            KeyError: If the appliance with the given ID does not exist in the database
        """
        # Get database instance
        db = self.db

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

        # Retrieve the appliance table from database
        appliance_table = getattr(db, 'appliance', None)

        # Check if appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if the appliance exists in the table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found")

        # Retrieve the appliance object
        appliance = appliance_table[appliance_id]

        # Get the lock status from the appliance
        # Note: lock_status is an optional field, so we need to handle None case
        lock_status = appliance.lock_status

        # If lock_status is None, it means the appliance doesn't support lock functionality
        # or the status hasn't been set yet
        if lock_status is None:
            raise KeyError(f"Appliance '{appliance_id}' does not have lock status information (lock functionality may not be supported)")

        # Return the lock status in the required format
        return {
            'lock_status': lock_status
        }

    @is_tool()
    def execute_appliance_operation_sequence(self, sequence_id: str) -> dict:
        """
        Execute a previously created operation sequence on an appliance.

        This method retrieves an operation sequence by its ID, executes all operations
        in the sequence on the specified appliance, and records the execution results.

        Args:
            sequence_id: Unique identifier of the sequence to execute

        Returns:
            dict: Contains 'success' (bool) and 'execution_results' (list of dicts)

        Raises:
            RuntimeError: If sequence not found, sequence already executed, or execution fails
        """
        import json
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the operation_sequence table
        operation_sequence_table = getattr(db, 'operation_sequence', None)
        if operation_sequence_table is None:
            raise RuntimeError("Operation sequence table not found in database")

        # Find the sequence by sequence_id
        sequence = operation_sequence_table.get(sequence_id)
        if sequence is None:
            raise RuntimeError(f"Operation sequence with ID '{sequence_id}' not found")

        # Check if the sequence has already been executed or is in progress
        if sequence.status == 'completed':
            raise RuntimeError(f"Operation sequence '{sequence_id}' has already been completed")
        elif sequence.status == 'executing':
            raise RuntimeError(f"Operation sequence '{sequence_id}' is currently being executed")
        elif sequence.status == 'failed':
            raise RuntimeError(f"Operation sequence '{sequence_id}' has failed previously")

        # Update sequence status to 'executing'
        sequence.status = 'executing'
        operation_sequence_table[sequence_id] = sequence
        setattr(db, 'operation_sequence', operation_sequence_table)

        # Parse the operations JSON array
        try:
            operations_list = json.loads(sequence.operations)
        except json.JSONDecodeError as e:
            # If parsing fails, mark sequence as failed
            sequence.status = 'failed'
            operation_sequence_table[sequence_id] = sequence
            setattr(db, 'operation_sequence', operation_sequence_table)
            raise RuntimeError(f"Failed to parse operations JSON: {str(e)}")

        # Validate that operations is a list
        if not isinstance(operations_list, list):
            sequence.status = 'failed'
            operation_sequence_table[sequence_id] = sequence
            setattr(db, 'operation_sequence', operation_sequence_table)
            raise RuntimeError("Operations must be a JSON array")

        # Execute each operation in sequence
        execution_results = []
        overall_success = True

        for operation in operations_list:
            # Each operation should be a dictionary with at least an 'operation' key
            if not isinstance(operation, dict):
                execution_results.append({
                    'operation': str(operation),
                    'success': False,
                    'error': 'Invalid operation format'
                })
                overall_success = False
                continue

            operation_name = operation.get('operation', 'unknown')

            try:
                # Here we simulate the execution of the operation
                # In a real implementation, this would call the actual appliance control methods
                # For this implementation, we assume all operations succeed
                # as we don't have access to the actual appliance control logic

                # Record successful execution
                execution_results.append({
                    'operation': operation_name,
                    'success': True
                })

            except Exception as e:
                # If any operation fails, record the failure
                execution_results.append({
                    'operation': operation_name,
                    'success': False,
                    'error': str(e)
                })
                overall_success = False

        # Update sequence status based on execution results
        current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
        if overall_success:
            sequence.status = 'completed'
        else:
            sequence.status = 'failed'

        sequence.executed_at = current_time

        # Save the updated sequence back to the database
        operation_sequence_table[sequence_id] = sequence
        setattr(db, 'operation_sequence', operation_sequence_table)

        # Return the execution results
        return {
            'success': overall_success,
            'execution_results': execution_results
        }

    @is_tool()
    def set_appliance_volume(self, appliance_id: str, volume_percentage: int) -> dict:
        """
        Set the volume level for appliances with audio output.

        Args:
            appliance_id: Unique identifier of the appliance
            volume_percentage: Volume level as percentage (0-100)

        Returns:
            dict: Contains success status and current volume percentage

        Raises:
            ValueError: If appliance not found, volume out of range, or appliance doesn't support volume control
        """
        # Validate volume_percentage is within valid range
        if not isinstance(volume_percentage, int):
            raise ValueError(f"volume_percentage must be an integer, got {type(volume_percentage).__name__}")

        if volume_percentage < 0 or volume_percentage > 100:
            raise ValueError(f"volume_percentage must be between 0 and 100, got {volume_percentage}")

        # Access the appliance table from database
        db = self.db
        appliance_table = getattr(db, 'appliance', None)

        if appliance_table is None:
            raise ValueError("Appliance table not found in database")

        # Check if appliance exists
        if appliance_id not in appliance_table:
            raise ValueError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Check if appliance supports volume control (volume_percentage field should not be None)
        if appliance.volume_percentage is None:
            raise ValueError(f"Appliance '{appliance_id}' does not support volume control")

        # Update the volume level by modifying the appliance object
        appliance.volume_percentage = volume_percentage

        # Write back the updated table to database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Return success response
        return {
            'success': True,
            'volume_percentage': volume_percentage
        }

    @is_tool()
    def send_custom_command(self, appliance_id: str, command_code: str, parameters: Optional[dict] = None) -> dict:
        """
        Send a custom command or instruction to an appliance for advanced control.

        This method validates the appliance exists and is connected, then executes
        a custom command with the provided parameters and updates the database state.

        Args:
            appliance_id: Unique identifier of the appliance
            command_code: Custom command code to execute
            parameters: Optional command parameters as key-value pairs

        Returns:
            dict: Result containing:
                - success (bool): Whether the command was executed successfully
                - response (dict): Response from the appliance with execution details

        Raises:
            ValueError: If appliance_id is empty, appliance not found, not connected,
                       or command_code is empty
        """
        from datetime import datetime

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

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

        # Ensure parameters is a dictionary if provided
        if parameters is None:
            parameters = {}
        elif not isinstance(parameters, dict):
            raise ValueError("parameters must be a dictionary")

        # Access the database
        db = self.db
        appliance_table = getattr(db, 'appliance', None)

        if appliance_table is None:
            raise ValueError("Appliance database table not found")

        # Check if appliance exists
        appliance = appliance_table.get(appliance_id)
        if appliance is None:
            raise ValueError(f"Appliance with ID '{appliance_id}' not found")

        # Verify appliance is connected (pre-condition)
        if not appliance.connected:
            raise ValueError(f"Appliance '{appliance_id}' is not connected to network")

        # Execute command and update database state
        current_time = datetime.now()

        # Update the appliance's last_seen timestamp to reflect recent activity
        appliance.last_seen = current_time

        # Apply command-specific updates based on command code and parameters
        # This handles common custom command patterns
        if command_code == "CMD_TURBO_MODE" or "turbo" in command_code.lower():
            # Turbo mode typically affects fan speed and operation status
            if hasattr(appliance, 'fan_speed'):
                appliance.fan_speed = "high"
            if hasattr(appliance, 'operation_status'):
                appliance.operation_status = "running"
        elif command_code == "CMD_QUIET_MODE" or "quiet" in command_code.lower():
            # Quiet mode typically sets low fan speed
            if hasattr(appliance, 'fan_speed'):
                appliance.fan_speed = "low"
        elif command_code == "CMD_AUTO_CLEAN" or "clean" in command_code.lower():
            # Cleaning mode updates operation status
            if hasattr(appliance, 'operation_status'):
                appliance.operation_status = "running"

        # Apply parameter-based updates if provided
        if parameters:
            # Handle intensity parameter
            if "intensity" in parameters:
                intensity = parameters["intensity"]
                if hasattr(appliance, 'fan_speed'):
                    if intensity == "high":
                        appliance.fan_speed = "high"
                    elif intensity == "medium":
                        appliance.fan_speed = "medium"
                    elif intensity == "low":
                        appliance.fan_speed = "low"

            # Handle mode parameter
            if "mode" in parameters and hasattr(appliance, 'mode'):
                appliance.mode = parameters["mode"]

            # Handle temperature parameter
            if "temperature" in parameters and hasattr(appliance, 'target_temperature'):
                try:
                    appliance.target_temperature = float(parameters["temperature"])
                except (ValueError, TypeError):
                    pass  # Skip invalid temperature values

        # Persist the updated appliance state to database
        appliance_table[appliance_id] = appliance

        # Build response based on command execution
        execution_time = current_time.strftime("%Y-%m-%d %H:%M:%S")

        response = {
            "status": "executed",
            "command_code": command_code,
            "appliance_id": appliance_id,
            "execution_time": execution_time,
            "parameters_applied": parameters
        }

        # Return success result with response (post-condition: command executed)
        return {
            "success": True,
            "response": response
        }

    @is_tool()
    def set_appliance_temperature(self, appliance_id: str, temperature: float) -> dict:
        """
        Set the target temperature for a temperature-controlled appliance like air conditioner or heater.

        Args:
            appliance_id: Unique identifier of the appliance
            temperature: Target temperature in Celsius

        Returns:
            dict: Contains success status and current temperature reading

        Raises:
            ValueError: If appliance not found, doesn't support temperature control, or is not powered on
        """
        # Validate input parameters
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

        if not isinstance(temperature, (int, float)):
            raise ValueError("temperature must be a number")

        # Access the database
        db = self.db

        # Get the appliance table
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise ValueError("Appliance table not found in database")

        # Check if appliance exists
        if appliance_id not in appliance_table:
            raise ValueError(f"Appliance with id '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Verify the appliance supports temperature control
        # Check if target_temperature field is available (not just exists as attribute, but is supported)
        if not hasattr(appliance, 'target_temperature'):
            raise ValueError(f"Appliance '{appliance_id}' does not support temperature control")

        # Verify the appliance is powered on (pre-condition check)
        # Use exact matching for power_state as it's a structured field with defined values
        if appliance.power_state != 'on':
            raise ValueError(f"Appliance '{appliance_id}' must be powered on to set temperature. Current state: {appliance.power_state}")

        # Set the target temperature
        appliance.target_temperature = float(temperature)

        # Get current temperature reading (may be None if sensor not available)
        current_temp = appliance.current_temperature if appliance.current_temperature is not None else 0.0

        # Update the appliance in the database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Return success response with current temperature
        return {
            'success': True,
            'current_temperature': float(current_temp)
        }

    @is_tool()
    def get_appliance_error_status(self, appliance_id: str) -> dict:
        """
        Retrieve current error or fault status of an appliance.

        This method queries the appliance_error table to fetch the error status
        information for the specified appliance. It returns whether the appliance
        has an error, along with error code and message if applicable.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing:
                - has_error (bool): Whether the appliance has an error
                - error_code (str): Error code if any (may be None)
                - error_message (str): Human-readable error message (may be None)

        Raises:
            KeyError: If the appliance_id is not found in the database
        """
        # Access the database instance
        db = self.db

        # Validate that appliance_id is provided and not empty
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

        # Get the appliance_error table from the database
        appliance_error_table = getattr(db, 'appliance_error', None)

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

        # Check if the appliance exists in the error table
        if appliance_id not in appliance_error_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found in the system")

        # Retrieve the error record for the specified appliance
        error_record = appliance_error_table[appliance_id]

        # Extract the error status information
        # has_error is a required field (boolean)
        has_error = error_record.has_error

        # error_code and error_message are optional fields
        # They may be None if no error exists or if not set
        error_code = error_record.error_code
        error_message = error_record.error_message

        # Construct and return the response dictionary
        return {
            'has_error': has_error,
            'error_code': error_code,
            'error_message': error_message
        }

    @is_tool()
    def pause_appliance_operation(self, appliance_id: str) -> dict:
        """
        Pause the current operation of an appliance such as washing machine or dishwasher.

        This method pauses an appliance that is currently in running state and supports pause functionality.
        It validates the appliance exists, is connected, powered on, and in running state before pausing.

        Args:
            appliance_id: Unique identifier of the appliance to pause

        Returns:
            dict: Contains success status and operation_status
                - success (bool): Whether the operation was paused successfully
                - operation_status (str): Current operation status after pause attempt

        Raises:
            RuntimeError: If appliance not found, not connected, not powered on, not in running state,
                         or does not support pause functionality
        """
        # Get database instance
        db = self.db

        # Validate input parameter - appliance_id must be non-empty string
        if not appliance_id or not isinstance(appliance_id, str):
            raise RuntimeError("Invalid appliance_id: must be a non-empty string")

        # Retrieve appliance table from database
        appliance_table = getattr(db, "appliance", None)
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if appliance exists in the database
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with id '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Validate pre-conditions: appliance must be connected
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        # Validate pre-conditions: appliance must be powered on
        if appliance.power_state not in ["on"]:
            raise RuntimeError(f"Appliance '{appliance_id}' is not powered on (current state: {appliance.power_state})")

        # Validate pre-conditions: appliance must be in running state
        if appliance.operation_status != "running":
            raise RuntimeError(f"Appliance '{appliance_id}' is not in running state (current status: {appliance.operation_status})")

        # Check if appliance supports pause functionality
        # Appliances that typically support pause: washing_machine, dishwasher, dryer, etc.
        pausable_appliance_types = [
            "washing_machine", 
            "dishwasher", 
            "dryer",
            "oven",
            "microwave",
            "robot_vacuum"
        ]

        if appliance.appliance_type not in pausable_appliance_types:
            raise RuntimeError(f"Appliance type '{appliance.appliance_type}' does not support pause functionality")

        # Pause the appliance operation by updating operation_status to 'paused'
        appliance.operation_status = "paused"

        # Update the appliance in the database
        appliance_table[appliance_id] = appliance
        setattr(db, "appliance", appliance_table)

        # Return success result with current operation status
        return {
            "success": True,
            "operation_status": "paused"
        }

    @is_tool()
    def get_appliance_usage_statistics(self, appliance_id: str, start_date: str, end_date: str) -> dict:
        """
        Retrieve usage statistics for an appliance over a specified time period.

        This method queries the appliance_usage_statistic table to find statistics
        that match the given appliance_id and overlap with the specified date range.

        Args:
            appliance_id: Unique identifier of the appliance
            start_date: Start date for statistics in yyyy-mm-dd format
            end_date: End date for statistics in yyyy-mm-dd format

        Returns:
            dict: Dictionary containing:
                - total_runtime_hours: Total runtime in hours
                - operation_count: Number of times the appliance was operated
                - total_energy_kwh: Total energy consumed in kilowatt-hours

        Raises:
            ValueError: If parameters are invalid or no statistics found
        """
        from datetime import datetime

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

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

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

        # Parse and validate date formats
        try:
            start_dt = datetime.strptime(start_date, "%Y-%m-%d").date()
        except ValueError:
            raise ValueError(f"start_date must be in yyyy-mm-dd format, got '{start_date}'")

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

        # Validate date range
        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
        appliance_usage_statistic_table = getattr(db, "appliance_usage_statistic", None)

        if appliance_usage_statistic_table is None:
            raise ValueError("appliance_usage_statistic table not found in database")

        # Find statistics records that match the appliance_id and overlap with the date range
        # The query period [start_dt, end_dt] overlaps with a record period [record.start_date, record.end_date] if:
        # record.start_date <= end_dt AND record.end_date >= start_dt
        matching_records = []
        for record_id, record in appliance_usage_statistic_table.items():
            if record.appliance_id == appliance_id:
                # Check if the record's date range overlaps with the query date range
                if record.start_date <= end_dt and record.end_date >= start_dt:
                    matching_records.append(record)

        # If no matching records found, raise an error
        if not matching_records:
            raise ValueError(f"No usage statistics found for appliance_id '{appliance_id}' in the period {start_date} to {end_date}")

        # Aggregate statistics from all matching records
        # Sum up all the statistics values
        total_runtime_hours = 0.0
        operation_count = 0
        total_energy_kwh = 0.0

        for record in matching_records:
            total_runtime_hours += float(record.total_runtime_hours)
            operation_count += int(record.operation_count)
            total_energy_kwh += float(record.total_energy_kwh)

        # Return the aggregated statistics
        return {
            "total_runtime_hours": total_runtime_hours,
            "operation_count": operation_count,
            "total_energy_kwh": total_energy_kwh
        }

    @is_tool()
    def get_appliance_volume(self, appliance_id: str) -> dict:
        """
        Retrieve the current volume level of an appliance.

        This method fetches the volume_percentage field from the appliance table
        for the specified appliance. The appliance must support volume control.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing volume_percentage (int)

        Raises:
            KeyError: If the appliance_id does not exist in the database
            ValueError: If the appliance does not support volume control
        """
        # Validate input parameter
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

        # Access the database
        db = self.db

        # Get the appliance table data
        appliance_table = getattr(db, 'appliance', None)

        # Check if the table exists
        if appliance_table is None:
            raise KeyError("Appliance table not found in database")

        # Check if the appliance exists in the table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with id '{appliance_id}' not found")

        # Retrieve the appliance object
        appliance = appliance_table[appliance_id]

        # Get the volume_percentage field
        volume_percentage = appliance.volume_percentage

        # Check pre-condition: appliance must support volume control
        # If volume_percentage is None, the appliance doesn't support volume control
        if volume_percentage is None:
            raise ValueError(f"Appliance '{appliance_id}' does not support volume control")

        # Ensure the volume_percentage is an integer
        if not isinstance(volume_percentage, int):
            try:
                volume_percentage = int(volume_percentage)
            except (ValueError, TypeError):
                raise ValueError(f"Invalid volume_percentage value for appliance '{appliance_id}'")

        # Return the volume information in the required format
        return {
            'volume_percentage': volume_percentage
        }

    @is_tool()
    def clear_appliance_error(self, appliance_id: str) -> dict:
        """
        Clear or reset an error state on an appliance after issue resolution.

        This method clears the error state for a specific appliance by:
        1. Verifying the appliance exists in the error table
        2. Checking that the appliance has an active error state
        3. Resetting the error flags and clearing error details
        4. Recording the time when the error was cleared

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: A dictionary containing:
                - success (bool): Whether the error was cleared successfully

        Raises:
            RuntimeError: If appliance not found or no active error state exists
        """
        from datetime import datetime

        # Input validation: check appliance_id is not empty
        if not appliance_id or not isinstance(appliance_id, str):
            raise RuntimeError("Invalid appliance_id: must be a non-empty string")

        # Access the database
        db = self.db

        # Retrieve the appliance_error table
        appliance_error_table = getattr(db, 'appliance_error', None)

        # Check if the table exists
        if appliance_error_table is None:
            raise RuntimeError(f"Appliance error table not found in database")

        # Check if the appliance exists in the error table
        if appliance_id not in appliance_error_table:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' not found in error table")

        # Retrieve the appliance error record
        error_record = appliance_error_table[appliance_id]

        # Pre-condition check: verify the appliance has an active error state
        if not error_record.has_error:
            raise RuntimeError(f"Appliance '{appliance_id}' does not have an active error state")

        # Clear the error state
        # Set has_error to False to indicate no error
        error_record.has_error = False

        # Clear error details
        error_record.error_code = None
        error_record.error_message = None

        # Record the time when the error was cleared in "yyyy-mm-dd HH:MM:SS" format
        error_record.error_cleared_at = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Note: error_occurred_at is kept to maintain history of when the error originally occurred

        # Update the database with the modified record
        appliance_error_table[appliance_id] = error_record
        setattr(db, 'appliance_error', appliance_error_table)

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

    @is_tool()
    def get_appliance_connection_status(self, appliance_id: str) -> dict:
        """
        Check if an appliance is currently connected to the network

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            Dictionary containing connection status and last seen time

        Raises:
            KeyError: If appliance with given ID is not found
        """
        # Get the database instance
        db = self.db

        # Retrieve the appliance table from the database
        appliance_table = getattr(db, 'appliance', None)

        # Check if the appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if the appliance_id exists in the table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found")

        # Retrieve the appliance record
        appliance = appliance_table[appliance_id]

        # Extract connection status and last seen time
        connected = appliance.connected
        last_seen = appliance.last_seen

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

        # Return the connection status information
        return {
            'connected': connected,
            'last_seen': last_seen_str
        }

    @is_tool()
    def get_appliance_temperature(self, appliance_id: str) -> dict:
        """
        Retrieve the current temperature setting and reading from a temperature-controlled appliance.

        This method fetches temperature data from the appliance database for a specific appliance.
        It validates that the appliance exists and returns both target and current temperature readings.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing:
                - target_temperature: Target temperature setting in Celsius (float or None)
                - current_temperature: Current temperature reading in Celsius (float or None)

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

        # Retrieve the appliance table from database
        appliance_table = getattr(db, 'appliance', None)

        # Check if appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if the appliance_id exists in the table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found")

        # Retrieve the appliance record
        appliance = appliance_table[appliance_id]

        # Extract temperature data from the appliance record
        # Both target_temperature and current_temperature are optional fields
        # They may be None if the appliance doesn't support temperature monitoring
        target_temperature = appliance.target_temperature
        current_temperature = appliance.current_temperature

        # Return the temperature data as a dictionary
        return {
            'target_temperature': target_temperature,
            'current_temperature': current_temperature
        }

    @is_tool()
    def set_appliance_swing_mode(self, appliance_id: str, swing_mode: str) -> dict:
        """
        Set the swing or oscillation mode for appliances like air conditioners or fans.

        Args:
            appliance_id: Unique identifier of the appliance
            swing_mode: Swing mode setting (horizontal, vertical, both, off)

        Returns:
            dict: Contains success status and current swing mode setting

        Raises:
            ValueError: If appliance not found, not connected, or doesn't support swing functionality
        """
        # Input validation - check swing_mode parameter
        if not swing_mode or not isinstance(swing_mode, str):
            raise ValueError("swing_mode must be a non-empty string")

        # Normalize swing_mode to lowercase for comparison
        swing_mode_normalized = swing_mode.strip().lower()
        valid_swing_modes = ['horizontal', 'vertical', 'both', 'off']

        if swing_mode_normalized not in valid_swing_modes:
            raise ValueError(f"Invalid swing_mode: {swing_mode}. Must be one of: {', '.join(valid_swing_modes)}")

        # Access database
        db = self.db
        appliance_table = getattr(db, 'appliance', None)

        if appliance_table is None:
            raise ValueError("Appliance database table not found")

        # Check if appliance exists
        appliance = appliance_table.get(appliance_id)
        if appliance is None:
            raise ValueError(f"Appliance with id '{appliance_id}' not found")

        # Pre-condition check: Verify appliance is connected
        if not appliance.connected:
            raise ValueError(f"Appliance '{appliance_id}' is not connected to network")

        # Pre-condition check: Verify appliance supports swing functionality
        # Appliances that typically support swing: air_conditioner, fan
        # Check if swing_mode field is available (not None initially means it's supported)
        appliance_type_lower = appliance.appliance_type.lower()
        swing_supported_types = ['air_conditioner', 'fan', 'air conditioner']

        # Check if appliance type supports swing mode
        supports_swing = any(supported_type in appliance_type_lower for supported_type in swing_supported_types)

        if not supports_swing:
            raise ValueError(f"Appliance type '{appliance.appliance_type}' does not support swing functionality")

        # Update the swing mode in the appliance record
        appliance.swing_mode = swing_mode_normalized

        # Save updated appliance back to database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Return success response with current swing mode
        return {
            'success': True,
            'swing_mode': swing_mode_normalized
        }

    @is_tool()
    def set_appliance_mode(self, appliance_id: str, mode: str) -> dict:
        """
        Set the operating mode for an appliance such as cooling, heating, fan, or auto mode.

        This method updates the operating mode of a specified appliance in the database.
        It validates that the appliance exists, is powered on, and supports the requested mode.

        Args:
            appliance_id: Unique identifier of the appliance
            mode: Operating mode to set (e.g., 'cooling', 'heating', 'fan', 'auto')

        Returns:
            dict: Contains 'success' (bool) and 'mode' (str) indicating the result

        Raises:
            ValueError: If appliance_id is invalid, appliance not found, not powered on,
                       doesn't support modes, or requested mode is not supported
        """
        # Input validation
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

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

        # Normalize mode to lowercase for consistent comparison
        mode = mode.strip().lower()

        # Access the database
        db = self.db

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

        # Check if appliance exists
        if appliance_id not in appliance_table:
            raise ValueError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance instance
        appliance = appliance_table[appliance_id]

        # Pre-condition: Check if appliance is powered on
        if appliance.power_state.lower() not in ['on']:
            raise ValueError(f"Appliance must be powered on to set mode. Current power state: {appliance.power_state}")

        # Check if appliance supports mode settings by verifying if mode field exists
        # and checking appliance features if available
        appliance_feature_table = getattr(db, 'appliance_feature', None)
        if appliance_feature_table is not None and appliance_id in appliance_feature_table:
            feature_entry = appliance_feature_table[appliance_id]

            # Check if available_modes is defined
            if feature_entry.available_modes:
                # Parse available modes (comma-separated list)
                available_modes = [m.strip().lower() for m in feature_entry.available_modes.split(',')]

                # Validate that the requested mode is supported
                if mode not in available_modes:
                    raise ValueError(
                        f"Mode '{mode}' is not supported for this appliance. "
                        f"Available modes: {', '.join(available_modes)}"
                    )

        # Update the appliance mode in the database
        appliance.mode = mode

        # Write back to database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Return success response with the current mode
        return {
            'success': True,
            'mode': mode
        }

    @is_tool()
    def get_appliance_supported_features(self, appliance_id: str) -> dict:
        """
        Retrieve a list of features and capabilities supported by an appliance.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing a list of supported features

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

        # Retrieve the appliance_feature table from the database
        appliance_feature_table = getattr(db, 'appliance_feature', None)

        # Check if the appliance_feature table exists
        if appliance_feature_table is None:
            raise KeyError(f"appliance_feature table not found in database")

        # Check if the appliance_id exists in the table
        if appliance_id not in appliance_feature_table:
            raise KeyError(f"Appliance with appliance_id '{appliance_id}' not found in the system")

        # Retrieve the appliance feature record
        appliance_feature_record = appliance_feature_table[appliance_id]

        # Extract the features string from the record
        features_str = appliance_feature_record.features

        # Parse the comma-separated features string into a list
        # Strip whitespace from each feature and filter out empty strings
        features_list = [feature.strip() for feature in features_str.split(',') if feature.strip()]

        # Return the features in the required format
        return {
            'features': features_list
        }

    @is_tool()
    def resume_appliance_operation(self, appliance_id: str) -> dict:
        """
        Resume a paused operation of an appliance

        This method resumes an appliance that is currently in paused state by:
        1. Validating the appliance exists in the database
        2. Checking that the appliance is currently in paused state
        3. Updating the operation status to 'running'
        4. Returning success status and current operation status

        Args:
            appliance_id: Unique identifier of the appliance to resume

        Returns:
            dict: Contains:
                - success (bool): Whether the operation was resumed successfully
                - operation_status (str): Current operation status after resume

        Raises:
            RuntimeError: If appliance doesn't exist, is not connected, or is not in paused state
        """
        from datetime import datetime

        # Input parameter validation
        if not appliance_id or not isinstance(appliance_id, str):
            raise RuntimeError("appliance_id must be a non-empty string")

        # Access the database
        db = self.db

        # Retrieve the appliance table
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise RuntimeError("Appliance database table not found")

        # Check if the appliance exists
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' does not exist")

        # Get the appliance record
        appliance = appliance_table[appliance_id]

        # Verify the appliance is connected to the network
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to the network")

        # Check pre-condition: appliance must be in paused state
        if appliance.operation_status != 'paused':
            raise RuntimeError(
                f"Cannot resume appliance '{appliance_id}': "
                f"current operation status is '{appliance.operation_status}', expected 'paused'"
            )

        # Resume the operation by updating the status to 'running'
        appliance.operation_status = 'running'

        # Update the appliance record in the database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Return success result with current operation status
        return {
            'success': True,
            'operation_status': 'running'
        }

    @is_tool()
    def set_appliance_brightness(self, appliance_id: str, brightness_percentage: int) -> dict:
        """
        Set the brightness level for appliances with lighting or display features

        Args:
            appliance_id: Unique identifier of the appliance
            brightness_percentage: Brightness level as percentage (0-100)

        Returns:
            dict: Contains success status and current brightness percentage

        Raises:
            ValueError: If appliance_id is invalid, appliance doesn't exist, 
                       brightness_percentage is out of range, or appliance doesn't support brightness
        """
        # Access the database
        db = self.db

        # Validate brightness_percentage is within valid range
        if not isinstance(brightness_percentage, int):
            raise ValueError("brightness_percentage must be an integer")

        if brightness_percentage < 0 or brightness_percentage > 100:
            raise ValueError("brightness_percentage must be between 0 and 100")

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

        # Retrieve appliance table from database
        appliance_table = getattr(db, "appliance", None)
        if appliance_table is None:
            raise ValueError("Appliance table not found in database")

        # Check if appliance exists in the database
        if appliance_id not in appliance_table:
            raise ValueError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Check if appliance supports brightness adjustment by checking appliance_feature table
        appliance_feature_table = getattr(db, "appliance_feature", None)
        if appliance_feature_table is not None and appliance_id in appliance_feature_table:
            feature_record = appliance_feature_table[appliance_id]
            features = feature_record.features if hasattr(feature_record, 'features') else ""
            # Check if brightness control is in the features list
            if features and isinstance(features, str):
                feature_list = [f.strip().lower() for f in features.split(',')]
                if 'brightness' not in feature_list and 'brightness_control' not in feature_list:
                    raise ValueError(f"Appliance '{appliance_id}' does not support brightness adjustment")

        # Update the brightness percentage
        appliance.brightness_percentage = brightness_percentage

        # Update the appliance in the database
        appliance_table[appliance_id] = appliance
        setattr(db, "appliance", appliance_table)

        # Return success response with current brightness level
        return {
            "success": True,
            "brightness_percentage": brightness_percentage
        }

    @is_tool()
    def get_appliance_swing_mode(self, appliance_id: str) -> dict:
        """
        Retrieve the current swing mode setting of an appliance.

        This method fetches the swing mode configuration from the appliance database.
        It validates that the appliance exists and returns its current swing mode setting.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing the swing_mode field with current setting

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

        # Retrieve the appliance table from the database
        appliance_table = getattr(db, "appliance", None)

        # Check if the appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if the appliance_id exists in the appliance table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found")

        # Retrieve the appliance object using the appliance_id
        appliance = appliance_table[appliance_id]

        # Extract the swing_mode attribute from the appliance object
        # The swing_mode can be None (optional field) or a string value
        swing_mode = appliance.swing_mode

        # Return the swing mode in the expected format
        # Note: Even if swing_mode is None, we return it as per the schema
        # The caller should handle None values if the appliance doesn't support swing
        return {
            "swing_mode": swing_mode
        }

    @is_tool()
    def calculate_appliance_energy_cost(self, energy_consumed_kwh: float, electricity_rate_per_kwh: float) -> dict:
        """
        Calculate the estimated energy cost for an appliance based on usage and electricity rate.

        This is a pure computation tool that does not require database access.
        It multiplies the energy consumed (in kWh) by the electricity rate (per kWh)
        to produce the estimated cost in local currency.

        Args:
            energy_consumed_kwh: Energy consumed in kilowatt-hours (must be non-negative)
            electricity_rate_per_kwh: Cost of electricity per kilowatt-hour in local currency (must be non-negative)

        Returns:
            dict: A dictionary containing:
                - estimated_cost (float): Estimated energy cost in local currency

        Raises:
            ValueError: If energy_consumed_kwh or electricity_rate_per_kwh is negative
            TypeError: If inputs are not numeric types
        """
        # Validate input types
        if not isinstance(energy_consumed_kwh, (int, float)):
            raise TypeError(f"energy_consumed_kwh must be a number, got {type(energy_consumed_kwh).__name__}")

        if not isinstance(electricity_rate_per_kwh, (int, float)):
            raise TypeError(f"electricity_rate_per_kwh must be a number, got {type(electricity_rate_per_kwh).__name__}")

        # Validate that values are non-negative
        if energy_consumed_kwh < 0:
            raise ValueError(f"energy_consumed_kwh must be non-negative, got {energy_consumed_kwh}")

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

        # Calculate the estimated cost
        # Formula: cost = energy_consumed (kWh) × rate (currency per kWh)
        estimated_cost = energy_consumed_kwh * electricity_rate_per_kwh

        # Round to 2 decimal places for currency precision
        estimated_cost = round(estimated_cost, 2)

        # Return the result as a dictionary
        return {
            "estimated_cost": estimated_cost
        }

    @is_tool()
    def validate_appliance_command(self, appliance_type: str, command: str, parameters: dict = None) -> dict:
        """
        Validate if a command is supported and appropriate for a specific appliance before execution.

        This method checks:
        1. Whether the command is a valid/known command for smart home appliances
        2. Whether the appliance type supports the given command
        3. Whether the provided parameters are valid for the command

        Args:
            appliance_type: Type of the appliance (e.g., 'air_conditioner', 'washing_machine')
            command: Command to validate (e.g., 'set_temperature', 'turn_on')
            parameters: Optional dictionary of parameters for the command

        Returns:
            dict: Validation result containing:
                - is_valid (bool): Whether the command is valid for this appliance
                - validation_message (str): Detailed validation result message

        Raises:
            ValueError: If appliance_type or command is empty/None
        """
        # Input validation
        if not appliance_type or not isinstance(appliance_type, str):
            raise ValueError("appliance_type must be a non-empty string")

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

        # Normalize inputs for comparison
        appliance_type = appliance_type.strip().lower()
        command = command.strip().lower()

        # Define comprehensive command-to-appliance-type mappings
        # Each command maps to a set of appliance types that support it
        command_support_map = {
            # Power control commands (universal)
            'turn_on': {'air_conditioner', 'washing_machine', 'light', 'tv', 'fan', 'heater', 'humidifier', 'dehumidifier', 'vacuum', 'dishwasher', 'oven', 'microwave', 'refrigerator', 'speaker', 'projector'},
            'turn_off': {'air_conditioner', 'washing_machine', 'light', 'tv', 'fan', 'heater', 'humidifier', 'dehumidifier', 'vacuum', 'dishwasher', 'oven', 'microwave', 'refrigerator', 'speaker', 'projector'},
            'toggle_power': {'air_conditioner', 'washing_machine', 'light', 'tv', 'fan', 'heater', 'humidifier', 'dehumidifier', 'vacuum', 'dishwasher', 'oven', 'microwave', 'refrigerator', 'speaker', 'projector'},

            # Temperature control commands
            'set_temperature': {'air_conditioner', 'heater', 'refrigerator', 'oven', 'water_heater'},
            'increase_temperature': {'air_conditioner', 'heater', 'oven', 'water_heater'},
            'decrease_temperature': {'air_conditioner', 'heater', 'oven', 'water_heater'},

            # Mode control commands
            'set_mode': {'air_conditioner', 'washing_machine', 'fan', 'heater', 'vacuum', 'dishwasher'},
            'change_mode': {'air_conditioner', 'washing_machine', 'fan', 'heater', 'vacuum', 'dishwasher'},

            # Fan/Speed control commands
            'set_fan_speed': {'air_conditioner', 'fan', 'heater'},
            'increase_fan_speed': {'air_conditioner', 'fan', 'heater'},
            'decrease_fan_speed': {'air_conditioner', 'fan', 'heater'},

            # Brightness control commands
            'set_brightness': {'light', 'tv', 'projector'},
            'increase_brightness': {'light', 'tv', 'projector'},
            'decrease_brightness': {'light', 'tv', 'projector'},

            # Volume control commands
            'set_volume': {'tv', 'speaker'},
            'increase_volume': {'tv', 'speaker'},
            'decrease_volume': {'tv', 'speaker'},
            'mute': {'tv', 'speaker'},
            'unmute': {'tv', 'speaker'},

            # Humidity control commands
            'set_humidity': {'humidifier', 'dehumidifier', 'air_conditioner'},
            'increase_humidity': {'humidifier'},
            'decrease_humidity': {'dehumidifier'},

            # Lock control commands
            'lock': {'washing_machine', 'dishwasher', 'oven'},
            'unlock': {'washing_machine', 'dishwasher', 'oven'},

            # Operation control commands
            'start': {'washing_machine', 'vacuum', 'dishwasher', 'oven', 'microwave'},
            'stop': {'washing_machine', 'vacuum', 'dishwasher', 'oven', 'microwave'},
            'pause': {'washing_machine', 'vacuum', 'dishwasher'},
            'resume': {'washing_machine', 'vacuum', 'dishwasher'},

            # Special feature commands
            'enable_eco_mode': {'air_conditioner', 'washing_machine', 'dishwasher', 'refrigerator'},
            'disable_eco_mode': {'air_conditioner', 'washing_machine', 'dishwasher', 'refrigerator'},
            'set_swing_mode': {'air_conditioner', 'fan'},
            'set_timer': {'air_conditioner', 'washing_machine', 'light', 'tv', 'fan', 'heater', 'humidifier', 'dehumidifier'},
            'cancel_timer': {'air_conditioner', 'washing_machine', 'light', 'tv', 'fan', 'heater', 'humidifier', 'dehumidifier'},
        }

        # Check if command is recognized
        if command not in command_support_map:
            return {
                'is_valid': False,
                'validation_message': f"Command '{command}' is not a recognized command"
            }

        # Check if appliance type supports the command
        supported_appliances = command_support_map[command]
        if appliance_type not in supported_appliances:
            return {
                'is_valid': False,
                'validation_message': f"Command '{command}' is not supported for appliance type '{appliance_type}'. Supported appliances: {', '.join(sorted(supported_appliances))}"
            }

        # Validate parameters if provided
        if parameters is not None:
            if not isinstance(parameters, dict):
                return {
                    'is_valid': False,
                    'validation_message': "Parameters must be a dictionary"
                }

            # Define parameter requirements for specific commands
            parameter_validation_rules = {
                'set_temperature': {
                    'required': ['temperature'],
                    'validators': {
                        'temperature': lambda v: isinstance(v, (int, float)) and -50 <= v <= 100
                    }
                },
                'set_fan_speed': {
                    'required': ['speed'],
                    'validators': {
                        'speed': lambda v: isinstance(v, str) and v.lower() in ['low', 'medium', 'high', 'auto']
                    }
                },
                'set_brightness': {
                    'required': ['brightness'],
                    'validators': {
                        'brightness': lambda v: isinstance(v, int) and 0 <= v <= 100
                    }
                },
                'set_volume': {
                    'required': ['volume'],
                    'validators': {
                        'volume': lambda v: isinstance(v, int) and 0 <= v <= 100
                    }
                },
                'set_humidity': {
                    'required': ['humidity'],
                    'validators': {
                        'humidity': lambda v: isinstance(v, int) and 0 <= v <= 100
                    }
                },
                'set_mode': {
                    'required': ['mode'],
                    'validators': {
                        'mode': lambda v: isinstance(v, str) and len(v) > 0
                    }
                },
                'set_swing_mode': {
                    'required': ['swing_mode'],
                    'validators': {
                        'swing_mode': lambda v: isinstance(v, str) and v.lower() in ['horizontal', 'vertical', 'both', 'off']
                    }
                },
                'set_timer': {
                    'required': ['duration_minutes'],
                    'validators': {
                        'duration_minutes': lambda v: isinstance(v, int) and v > 0
                    }
                }
            }

            # Check if command has parameter requirements
            if command in parameter_validation_rules:
                rules = parameter_validation_rules[command]

                # Check required parameters
                for required_param in rules['required']:
                    if required_param not in parameters:
                        return {
                            'is_valid': False,
                            'validation_message': f"Command '{command}' requires parameter '{required_param}'"
                        }

                # Validate parameter values
                for param_name, validator_func in rules['validators'].items():
                    if param_name in parameters:
                        if not validator_func(parameters[param_name]):
                            return {
                                'is_valid': False,
                                'validation_message': f"Invalid value for parameter '{param_name}' in command '{command}'"
                            }

        # All validations passed
        return {
            'is_valid': True,
            'validation_message': 'Command is supported and parameters are valid'
        }

    @is_tool()
    def get_appliance_brightness(self, appliance_id: str) -> dict:
        """
        Retrieve the current brightness level of an appliance.

        This method fetches the brightness percentage of a specified appliance from the database.
        It validates that the appliance exists and supports brightness control.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing brightness_percentage (int) - Current brightness level percentage (0-100)

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

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

        # Retrieve the appliance table from database
        appliance_table = getattr(db, "appliance", None)

        # Check if appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if the appliance_id exists in the table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with appliance_id '{appliance_id}' not found")

        # Retrieve the appliance object
        appliance = appliance_table[appliance_id]

        # Get the brightness_percentage attribute
        # Note: brightness_percentage is Optional[int] in schema, so it may be None
        brightness = appliance.brightness_percentage

        # Check if brightness control is supported (brightness_percentage should not be None)
        # According to pre_condition: "Appliance must support brightness control"
        if brightness is None:
            raise ValueError(f"Appliance '{appliance_id}' does not support brightness control")

        # Return the brightness information as specified in the returns schema
        return {
            "brightness_percentage": brightness
        }

    @is_tool()
    def get_appliance_timer_status(self, appliance_id: str) -> dict:
        """
        Retrieve the current timer status and remaining time for an appliance.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Timer status information containing:
                - timer_active: Whether a timer is currently active
                - remaining_minutes: Remaining time in minutes
                - timer_action: Action to be performed when timer expires

        Raises:
            KeyError: If the appliance_id is not found in the appliance_timer table
        """
        # Access the database instance
        db = self.db

        # Retrieve the appliance_timer table from the database
        appliance_timer_table = getattr(db, 'appliance_timer', None)

        # Check if the table exists
        if appliance_timer_table is None:
            raise KeyError(f"appliance_timer table not found in database")

        # Check if the appliance_id exists in the table
        if appliance_id not in appliance_timer_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found in appliance_timer table")

        # Retrieve the timer record for the specified appliance
        timer_record = appliance_timer_table[appliance_id]

        # Extract the required fields from the timer record
        # timer_active is a required field indicating if timer is currently active
        timer_active = timer_record.timer_active

        # remaining_minutes may be None if no timer is active
        remaining_minutes = timer_record.remaining_minutes

        # timer_action may be None if no timer is active
        timer_action = timer_record.timer_action

        # Construct the return dictionary with the timer status information
        result = {
            'timer_active': timer_active,
            'remaining_minutes': remaining_minutes,
            'timer_action': timer_action
        }

        return result

    @is_tool()
    def cancel_appliance_timer(self, appliance_id: str) -> dict:
        """
        Cancel an active timer on an appliance.

        This method cancels the active timer for the specified appliance by updating
        the timer_active status to False and clearing all timer-related fields.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: A dictionary containing:
                - success (bool): Whether the timer was cancelled successfully

        Raises:
            RuntimeError: If the appliance does not exist, has no timer, or the timer is not active
        """
        # Access the database
        db = self.db

        # Retrieve the appliance_timer table
        appliance_timer_table = getattr(db, "appliance_timer", None)

        # Check if the appliance_timer table exists
        if appliance_timer_table is None:
            raise RuntimeError(f"Appliance timer table does not exist in the database")

        # Check if the appliance exists in the timer table
        timer_record = appliance_timer_table.get(appliance_id)
        if timer_record is None:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' does not have a timer record")

        # Check if the timer is currently active
        if not timer_record.timer_active:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' does not have an active timer")

        # Cancel the timer by setting timer_active to False and clearing timer fields
        timer_record.timer_active = False
        timer_record.duration_minutes = None
        timer_record.remaining_minutes = None
        timer_record.timer_action = None
        timer_record.timer_end_time = None
        timer_record.started_at = None

        # Update the database with the modified timer record
        appliance_timer_table[appliance_id] = timer_record
        setattr(db, "appliance_timer", appliance_timer_table)

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

    @is_tool()
    def get_appliance_firmware_version(self, appliance_id: str):
        """
        Retrieve the current firmware version of an appliance.

        This method fetches the firmware version and last update date for a specific appliance
        from the database. The appliance must be connected to the network.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing:
                - firmware_version: Current firmware version string
                - last_updated: Date of last firmware update in 'yyyy-mm-dd HH:MM:SS' format

        Raises:
            KeyError: If the appliance_id does not exist in the database
        """
        # Validate input parameter
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

        # Access the database
        db = self.db

        # Get the appliance table from database
        appliance_table = getattr(db, 'appliance', None)

        # Check if appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Retrieve the specific appliance by ID
        # The table is stored as Dict[str, Appliance] where key is appliance_id
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found")

        appliance = appliance_table[appliance_id]

        # Extract firmware version information
        firmware_version = appliance.firmware_version
        firmware_updated_at = appliance.firmware_updated_at

        # Convert datetime object to required string format 'yyyy-mm-dd HH:MM:SS'
        from datetime import datetime
        if isinstance(firmware_updated_at, datetime):
            last_updated_str = firmware_updated_at.strftime("%Y-%m-%d %H:%M:%S")
        else:
            # If it's already a string, ensure it's in the correct format
            last_updated_str = str(firmware_updated_at)

        # Return the firmware information as specified in the schema
        return {
            'firmware_version': firmware_version,
            'last_updated': last_updated_str
        }

    @is_tool()
    def convert_temperature_unit(self, temperature: float, from_unit: str, to_unit: str) -> dict:
        """
        Convert temperature values between Celsius and Fahrenheit for appliance control.

        This method performs temperature unit conversion without requiring database access.
        It validates input parameters and applies the appropriate conversion formula.

        Args:
            temperature: Temperature value to convert (numeric)
            from_unit: Source temperature unit (celsius, fahrenheit)
            to_unit: Target temperature unit (celsius, fahrenheit)

        Returns:
            dict: Contains 'converted_temperature' key with the converted value

        Raises:
            ValueError: If units are invalid or if conversion is not possible
            TypeError: If temperature is not a numeric type
        """
        # Validate temperature parameter type
        if not isinstance(temperature, (int, float)):
            raise TypeError(f"Temperature must be a numeric value, got {type(temperature).__name__}")

        # Normalize unit strings to lowercase for case-insensitive comparison
        from_unit_normalized = from_unit.lower().strip()
        to_unit_normalized = to_unit.lower().strip()

        # Define valid temperature units
        valid_units = ['celsius', 'fahrenheit']

        # Validate from_unit
        if from_unit_normalized not in valid_units:
            raise ValueError(
                f"Invalid source unit '{from_unit}'. Must be one of: {', '.join(valid_units)}"
            )

        # Validate to_unit
        if to_unit_normalized not in valid_units:
            raise ValueError(
                f"Invalid target unit '{to_unit}'. Must be one of: {', '.join(valid_units)}"
            )

        # If units are the same, return the original temperature
        if from_unit_normalized == to_unit_normalized:
            return {
                'converted_temperature': temperature
            }

        # Perform conversion based on unit combination
        if from_unit_normalized == 'celsius' and to_unit_normalized == 'fahrenheit':
            # Celsius to Fahrenheit: F = (C × 9/5) + 32
            converted_temperature = (temperature * 9/5) + 32
        elif from_unit_normalized == 'fahrenheit' and to_unit_normalized == 'celsius':
            # Fahrenheit to Celsius: C = (F - 32) × 5/9
            converted_temperature = (temperature - 32) * 5/9
        else:
            # This should never happen due to validation above, but included for completeness
            raise ValueError(
                f"Unsupported conversion from '{from_unit}' to '{to_unit}'"
            )

        # Round to one decimal place for practical temperature precision
        converted_temperature = round(converted_temperature, 1)

        return {
            'converted_temperature': converted_temperature
        }

    @is_tool()
    def get_appliance_mode(self, appliance_id: str) -> dict:
        """
        Retrieve the current operating mode of an appliance

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing:
                - mode: Current operating mode
                - available_modes: List of all available operating modes

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

        # Retrieve appliance table data
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if appliance exists in the appliance table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance record
        appliance = appliance_table[appliance_id]

        # Extract the current mode from appliance record
        current_mode = appliance.mode

        # Retrieve appliance_feature table data to get available modes
        appliance_feature_table = getattr(db, 'appliance_feature', None)
        if appliance_feature_table is None:
            raise KeyError(f"Appliance feature table not found in database")

        # Check if appliance has feature record
        if appliance_id not in appliance_feature_table:
            raise KeyError(f"Appliance feature record for ID '{appliance_id}' not found")

        # Get the appliance feature record
        appliance_feature = appliance_feature_table[appliance_id]

        # Parse available_modes from comma-separated string to list
        available_modes_list = []
        if appliance_feature.available_modes:
            # Split by comma and strip whitespace from each mode
            available_modes_list = [mode.strip() for mode in appliance_feature.available_modes.split(',')]

        # Construct return dictionary
        result = {
            'mode': current_mode,
            'available_modes': available_modes_list
        }

        return result

    @is_tool()
    def stop_appliance_operation(self, appliance_id: str) -> dict:
        """
        Stop the current operation of an appliance completely.

        This method stops the appliance operation by updating its operation_status to 'stopped'.
        It validates that the appliance exists, is connected, and is in a valid state (running or paused)
        before stopping the operation.

        Args:
            appliance_id: Unique identifier of the appliance to stop

        Returns:
            dict: Contains success status and current operation status
                - success (bool): Whether the operation was stopped successfully
                - operation_status (str): Current operation status after stopping

        Raises:
            ValueError: If appliance_id is empty or None
            RuntimeError: If appliance not found, not connected, or not in valid state to stop
        """
        from datetime import datetime

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

        # Access the database
        db = self.db

        # Retrieve the appliance table
        appliance_table = getattr(db, "appliance", None)
        if appliance_table is None:
            raise RuntimeError(f"Appliance table not found in database")

        # Check if the appliance exists
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance instance
        appliance = appliance_table[appliance_id]

        # Verify the appliance is connected
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        # Check pre-condition: appliance must be in running or paused state
        current_status = appliance.operation_status
        if current_status not in ["running", "paused"]:
            raise RuntimeError(
                f"Appliance '{appliance_id}' cannot be stopped. "
                f"Current operation status is '{current_status}'. "
                f"Only appliances in 'running' or 'paused' state can be stopped."
            )

        # Stop the operation by updating the operation_status
        appliance.operation_status = "stopped"

        # Update the appliance in the database
        appliance_table[appliance_id] = appliance
        setattr(db, "appliance", appliance_table)

        # Return success response with the new operation status
        return {
            "success": True,
            "operation_status": "stopped"
        }

    @is_tool()
    def enable_appliance_notifications(self, appliance_id: str, notification_types: list) -> dict:
        """
        Enable push notifications for appliance status changes and alerts.

        This method enables specified notification types for a given appliance by creating
        or updating the notification settings in the database.

        Args:
            appliance_id: Unique identifier of the appliance
            notification_types: List of notification types to enable (e.g., ['operation_complete', 'error_alert'])

        Returns:
            dict: Contains 'success' boolean indicating if notifications were enabled successfully

        Raises:
            ValueError: If appliance_id is empty, notification_types is empty, or appliance doesn't exist
        """
        from datetime import datetime

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

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

        if not all(isinstance(nt, str) for nt in notification_types):
            raise ValueError("All notification types must be strings")

        # Access the database
        db = self.db

        # Get the appliance_notification_setting table
        notification_settings = getattr(db, 'appliance_notification_setting', None)

        # Initialize the table if it doesn't exist
        if notification_settings is None:
            notification_settings = {}
            setattr(db, 'appliance_notification_setting', notification_settings)

        # Convert notification_types list to comma-separated string
        enabled_types_str = ','.join(notification_types)

        # Get current timestamp in required format
        current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Check if notification setting already exists for this appliance
        existing_setting = notification_settings.get(appliance_id)

        if existing_setting:
            # Update existing notification setting
            # Merge new notification types with existing ones to avoid duplicates
            existing_types = set()
            if existing_setting.enabled_notification_types:
                existing_types = set(existing_setting.enabled_notification_types.split(','))

            # Add new notification types
            existing_types.update(notification_types)

            # Update the setting
            existing_setting.notifications_enabled = True
            existing_setting.enabled_notification_types = ','.join(sorted(existing_types))
            existing_setting.updated_at = current_time
        else:
            # Create new notification setting record
            new_setting = ApplianceNotificationSetting(

                appliance_id=appliance_id,

                notifications_enabled=True,

                enabled_notification_types=enabled_types_str,

                updated_at=current_time

            )

            # Add to the table
            notification_settings[appliance_id] = new_setting

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

    @is_tool()
    def reset_appliance(self, appliance_id: str) -> dict:
        """
        Perform a soft reset of an appliance to restore default settings or clear temporary states.

        Args:
            appliance_id: Unique identifier of the appliance to reset

        Returns:
            dict: Contains 'success' boolean indicating whether the reset was successful

        Raises:
            RuntimeError: If appliance is not found, not powered on, or reset operation fails
        """
        from datetime import datetime

        # Input validation - check appliance_id is not empty
        if not appliance_id or not isinstance(appliance_id, str):
            raise RuntimeError("Invalid appliance_id: must be a non-empty string")

        # Access the database
        db = self.db

        # Get the appliance table
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if appliance exists
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with id '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Pre-condition: Appliance must be powered on
        if appliance.power_state != 'on':
            raise RuntimeError(f"Appliance '{appliance_id}' must be powered on to perform reset. Current state: {appliance.power_state}")

        # Check if appliance is connected to network
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        try:
            # Perform soft reset by restoring default settings
            current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

            # Reset temperature-related settings (for air conditioners, etc.)
            if appliance.target_temperature is not None:
                appliance.target_temperature = None

            # Reset mode to default (if applicable)
            if appliance.mode is not None:
                appliance.mode = None

            # Reset fan speed to default (if applicable)
            if appliance.fan_speed is not None:
                appliance.fan_speed = None

            # Reset brightness to default (for lights, displays, etc.)
            if appliance.brightness_percentage is not None:
                appliance.brightness_percentage = None

            # Reset volume and mute status (for audio devices, TVs, etc.)
            if appliance.volume_percentage is not None:
                appliance.volume_percentage = None
            if appliance.mute_status is not None:
                appliance.mute_status = None

            # Reset lock status (for doors, appliances with child lock, etc.)
            if appliance.lock_status is not None:
                appliance.lock_status = None

            # Reset operation status to idle and clear progress
            appliance.operation_status = 'idle'
            if appliance.progress_percentage is not None:
                appliance.progress_percentage = None

            # Reset eco mode to disabled (default state)
            appliance.eco_mode_enabled = False

            # Reset swing mode (for air conditioners, fans, etc.)
            if appliance.swing_mode is not None:
                appliance.swing_mode = None

            # Reset humidity settings (for humidifiers, dehumidifiers, etc.)
            if appliance.target_humidity is not None:
                appliance.target_humidity = None

            # Update last seen time to reflect the reset operation - must use string format
            appliance.last_seen = current_time

            # Update the appliance in the database
            appliance_table[appliance_id] = appliance
            setattr(db, 'appliance', appliance_table)

            # Return success
            return {'success': True}

        except Exception as e:
            # If any error occurs during reset, raise RuntimeError
            raise RuntimeError(f"Failed to reset appliance '{appliance_id}': {str(e)}")

    @is_tool()
    def mute_appliance(self, appliance_id: str) -> dict:
        """
        Mute the sound output or beep notifications of an appliance.

        This method updates the mute_status field of the specified appliance to 'muted'.
        It validates that the appliance exists and is connected before performing the operation.

        Args:
            appliance_id: Unique identifier of the appliance to mute

        Returns:
            dict: Contains success status and current mute status
                - success (bool): Whether the appliance was muted successfully
                - mute_status (str): Current mute status after operation

        Raises:
            RuntimeError: If appliance not found, not connected, or operation fails
        """
        # Input validation - check appliance_id is not empty
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("Invalid appliance_id: must be a non-empty string")

        # Access the database
        db = self.db

        # Retrieve the appliance table
        appliance_table = getattr(db, "appliance", None)
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if the appliance exists
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Verify the appliance is connected to the network
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        # Check if the appliance supports sound output (mute_status field)
        # If mute_status is None, it means the appliance doesn't support sound output
        if not hasattr(appliance, 'mute_status') or appliance.mute_status is None:
            raise RuntimeError(f"Appliance '{appliance_id}' does not support sound output")

        # Update the mute status to 'muted'
        appliance.mute_status = "muted"

        # Update the appliance in the database
        appliance_table[appliance_id] = appliance
        setattr(db, "appliance", appliance_table)

        # Return success response with current mute status
        return {
            "success": True,
            "mute_status": "muted"
        }

    @is_tool()
    def set_appliance_fan_speed(self, appliance_id: str, fan_speed: str) -> dict:
        """
        Set the fan speed level for appliances with adjustable fans.

        This method updates the fan speed setting for a specified appliance in the database.
        It validates that the appliance exists, is powered on, supports fan speed adjustment,
        and that the requested fan speed value is valid.

        Args:
            appliance_id: Unique identifier of the appliance
            fan_speed: Fan speed level (low, medium, high, auto)

        Returns:
            dict: Contains success status and current fan speed setting
                - success (bool): Whether the fan speed was set successfully
                - fan_speed (str): Current fan speed setting after update

        Raises:
            ValueError: If appliance not found, not powered on, doesn't support fan speed,
                       or invalid fan speed value provided
        """
        # Validate input parameters
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

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

        # Normalize fan_speed to lowercase for comparison
        fan_speed_normalized = fan_speed.strip().lower()

        # Define valid fan speed values
        valid_fan_speeds = ["low", "medium", "high", "auto"]

        # Validate fan_speed value
        if fan_speed_normalized not in valid_fan_speeds:
            raise ValueError(f"Invalid fan_speed '{fan_speed}'. Must be one of: {', '.join(valid_fan_speeds)}")

        # Access the database
        db = self.db

        # Retrieve the appliance table
        appliance_table = getattr(db, "appliance", None)
        if appliance_table is None:
            raise ValueError("Appliance table not found in database")

        # Check if appliance exists
        if appliance_id not in appliance_table:
            raise ValueError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Verify appliance is powered on (pre-condition)
        if appliance.power_state.lower() != "on":
            raise ValueError(f"Appliance '{appliance_id}' must be powered on to adjust fan speed. Current state: {appliance.power_state}")

        # Verify appliance supports fan speed adjustment
        # Check if fan_speed field is available (not None) in the database schema
        current_fan_speed = getattr(appliance, 'fan_speed', None)
        if current_fan_speed is None:
            raise ValueError(f"Appliance '{appliance_id}' does not support fan speed adjustment")

        # Update the fan speed in the appliance object
        appliance.fan_speed = fan_speed_normalized

        # Write the updated appliance back to the database
        appliance_table[appliance_id] = appliance
        setattr(db, "appliance", appliance_table)

        # Return success response with the updated fan speed
        return {
            "success": True,
            "fan_speed": fan_speed_normalized
        }

    @is_tool()
    def get_appliance_fan_speed(self, appliance_id: str):
        """
        Retrieve the current fan speed setting of an appliance

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing the fan_speed field

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

        # Get the appliance table from the database
        appliance_table = getattr(db, 'appliance', None)

        # Check if the appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if the appliance_id exists in the table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found")

        # Retrieve the appliance record
        appliance = appliance_table[appliance_id]

        # Get the fan_speed attribute from the appliance
        # Note: fan_speed is an optional field, so it might be None
        fan_speed = appliance.fan_speed

        # Return the fan speed in the expected format
        return {
            'fan_speed': fan_speed
        }

    @is_tool()
    def estimate_operation_duration(self, appliance_type: str, mode: str, load_size: str = None) -> dict:
        """
        Estimate the duration required for an appliance to complete a specific operation.

        This method calculates estimated operation duration based on appliance type, operation mode,
        and optional load size. The estimation uses predefined duration mappings for common appliance
        types and their operation modes.

        Args:
            appliance_type: Type of the appliance (e.g., 'washing_machine', 'dishwasher', 'dryer')
            mode: Operation mode or program (e.g., 'quick_wash', 'normal', 'heavy_duty')
            load_size: Size of the load - 'small', 'medium', or 'large' (optional)

        Returns:
            dict: Dictionary containing 'estimated_duration_minutes' (integer)

        Raises:
            ValueError: If appliance_type or mode is empty/invalid, or if unsupported values are provided
        """

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

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

        # Normalize inputs to lowercase for case-insensitive matching
        appliance_type = appliance_type.strip().lower()
        mode = mode.strip().lower()

        # Validate and normalize load_size if provided
        if load_size is not None:
            if not isinstance(load_size, str) or not load_size.strip():
                raise ValueError("load_size must be a non-empty string if provided")
            load_size = load_size.strip().lower()
            if load_size not in ['small', 'medium', 'large']:
                raise ValueError("load_size must be one of: 'small', 'medium', 'large'")

        # Duration estimation mappings for different appliance types and modes
        # Base durations are defined for each appliance_type and mode combination
        duration_mappings = {
            'washing_machine': {
                'quick_wash': 30,
                'normal': 60,
                'heavy_duty': 90,
                'delicate': 45,
                'eco': 120,
                'spin_only': 15,
                'rinse_spin': 25
            },
            'dishwasher': {
                'quick_wash': 45,
                'normal': 90,
                'heavy_duty': 150,
                'eco': 180,
                'rinse_only': 15,
                'sanitize': 120
            },
            'dryer': {
                'quick_dry': 30,
                'normal': 60,
                'heavy_duty': 90,
                'delicate': 45,
                'air_dry': 120,
                'steam_refresh': 20
            },
            'oven': {
                'preheat': 15,
                'bake': 45,
                'roast': 90,
                'broil': 30,
                'convection': 40,
                'self_clean': 180
            },
            'air_conditioner': {
                'cool': 30,
                'heat': 30,
                'fan': 60,
                'auto': 45,
                'dry': 60,
                'eco': 90
            },
            'vacuum_cleaner': {
                'quick_clean': 15,
                'normal': 30,
                'deep_clean': 60,
                'spot_clean': 10,
                'auto': 45
            },
            'robot_vacuum': {
                'quick_clean': 20,
                'normal': 45,
                'deep_clean': 90,
                'spot_clean': 10,
                'edge_clean': 30,
                'auto': 60
            }
        }

        # Load size adjustment factors (multipliers for base duration)
        load_size_factors = {
            'small': 0.75,   # 25% reduction
            'medium': 1.0,   # No change (base duration)
            'large': 1.3     # 30% increase
        }

        # Check if appliance_type is supported
        if appliance_type not in duration_mappings:
            raise ValueError(
                f"Unsupported appliance_type: '{appliance_type}'. "
                f"Supported types: {', '.join(duration_mappings.keys())}"
            )

        # Check if mode is supported for this appliance type
        if mode not in duration_mappings[appliance_type]:
            raise ValueError(
                f"Unsupported mode '{mode}' for appliance_type '{appliance_type}'. "
                f"Supported modes: {', '.join(duration_mappings[appliance_type].keys())}"
            )

        # Get base duration for the appliance type and mode
        base_duration = duration_mappings[appliance_type][mode]

        # Apply load size adjustment if provided
        if load_size is not None:
            adjustment_factor = load_size_factors[load_size]
            estimated_duration = int(base_duration * adjustment_factor)
        else:
            # If no load_size provided, use base duration (equivalent to 'medium')
            estimated_duration = base_duration

        # Ensure minimum duration of 1 minute
        estimated_duration = max(1, estimated_duration)

        return {
            'estimated_duration_minutes': estimated_duration
        }

    @is_tool()
    def check_appliance_firmware_update(self, appliance_id: str) -> dict:
        """
        Check if a firmware update is available for an appliance.

        This method retrieves the appliance's current firmware version and simulates
        checking for available updates by comparing against a mock latest version.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Contains update_available (bool) and latest_version (str)

        Raises:
            RuntimeError: If appliance is not found or not connected to network
        """
        # Input parameter validation
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

        # Access the database
        db = self.db

        # Get appliance table data
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise RuntimeError("Appliance database table not found")

        # Check if appliance exists
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' not found")

        # Retrieve the appliance object
        appliance = appliance_table[appliance_id]

        # Verify pre-condition: appliance must be connected to network
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        # Get current firmware version
        current_version = appliance.firmware_version

        # Simulate firmware update check logic
        # Parse version string (assuming format like "1.2.3" or "2.5.0")
        # In a real system, this would query an external firmware update service
        try:
            # Parse current version into components
            current_parts = [int(x) for x in current_version.split('.')]
        except (ValueError, AttributeError):
            # If version format is unexpected, assume no update available
            return {
                'update_available': False,
                'latest_version': current_version
            }

        # Simulate latest version determination
        # For demonstration, we increment the minor version by 1 to simulate a newer version
        # In production, this would come from a firmware update server
        latest_parts = current_parts.copy()

        # Simulate update availability logic:
        # - If current version is older than 2.5.0, update is available to 2.5.0
        # - Otherwise, no update available (already on latest)
        reference_version = [2, 5, 0]

        # Compare versions
        update_available = False
        latest_version = current_version

        # Pad versions to same length for comparison
        max_len = max(len(current_parts), len(reference_version))
        current_padded = current_parts + [0] * (max_len - len(current_parts))
        reference_padded = reference_version + [0] * (max_len - len(reference_version))

        # Check if current version is older than reference version
        for i in range(max_len):
            if current_padded[i] < reference_padded[i]:
                update_available = True
                latest_version = '.'.join(map(str, reference_version))
                break
            elif current_padded[i] > reference_padded[i]:
                # Current version is newer than reference
                break

        # Return firmware update status
        return {
            'update_available': update_available,
            'latest_version': latest_version
        }

    @is_tool()
    def batch_control_appliances(self, appliance_ids: list, command: str) -> dict:
        """
        Control multiple appliances simultaneously with a single command.

        This method executes a command (turn_on or turn_off) on multiple appliances
        and returns the execution results for each appliance.

        Args:
            appliance_ids: List of appliance identifiers to control
            command: Command to execute (turn_on, turn_off)

        Returns:
            dict: Dictionary containing results list with success status for each appliance

        Raises:
            ValueError: If appliance_ids is empty or command is invalid
            RuntimeError: If appliances cannot be controlled or database operation fails
        """
        from datetime import datetime

        # Validate input parameters
        if not appliance_ids:
            raise ValueError("appliance_ids list cannot be empty")

        if not isinstance(appliance_ids, list):
            raise ValueError("appliance_ids must be a list")

        # Validate command parameter
        valid_commands = ['turn_on', 'turn_off']
        if command not in valid_commands:
            raise ValueError(f"Invalid command '{command}'. Must be one of: {', '.join(valid_commands)}")

        # Access the database
        db = self.db
        appliance_table = getattr(db, 'appliance', None)

        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Determine the target power state based on command
        if command == 'turn_on':
            target_power_state = 'on'
        elif command == 'turn_off':
            target_power_state = 'off'

        # Pre-validate all appliances meet the pre-condition
        for appliance_id in appliance_ids:
            appliance = appliance_table.get(appliance_id)

            if appliance is None:
                raise RuntimeError(f"Appliance '{appliance_id}' not found")

            if not appliance.connected:
                raise RuntimeError(f"Appliance '{appliance_id}' is not connected")

        # Initialize results list
        results = []
        current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Process each appliance
        for appliance_id in appliance_ids:
            result = {
                'appliance_id': appliance_id,
                'success': False
            }

            try:
                # Get appliance (we already validated it exists)
                appliance = appliance_table.get(appliance_id)

                # Update the appliance power state
                appliance.power_state = target_power_state
                appliance.power_state_updated_at = current_time

                # Update the appliance in database
                appliance_table[appliance_id] = appliance

                # Mark as successful
                result['success'] = True

            except Exception as e:
                # If any error occurs during processing this appliance, raise RuntimeError
                raise RuntimeError(f"Failed to control appliance '{appliance_id}': {str(e)}")

            results.append(result)

        return {
            'results': results
        }

    @is_tool()
    def unmute_appliance(self, appliance_id: str) -> dict:
        """
        Unmute the sound output or beep notifications of an appliance.

        Args:
            appliance_id: Unique identifier of the appliance to unmute

        Returns:
            dict: Contains success status and current mute status

        Raises:
            RuntimeError: If appliance not found, not connected, or not in muted state
        """
        # Input validation: check if appliance_id is provided and valid
        if not appliance_id or not isinstance(appliance_id, str):
            raise RuntimeError("Invalid appliance_id: must be a non-empty string")

        # Access the database
        db = self.db

        # Retrieve the appliance table from database
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if the appliance exists in the database
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Verify appliance is connected to network
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        # Check pre-condition: appliance must be in muted state
        if appliance.mute_status != 'muted':
            raise RuntimeError(
                f"Appliance '{appliance_id}' is not in muted state. "
                f"Current mute status: {appliance.mute_status}"
            )

        # Unmute the appliance by updating mute_status to 'unmuted'
        appliance.mute_status = 'unmuted'

        # Update the appliance in the database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Return success response with current mute status
        return {
            'success': True,
            'mute_status': 'unmuted'
        }

    @is_tool()
    def turn_off_appliance(self, appliance_id: str) -> dict:
        """
        Turn off a specific smart home appliance remotely.

        This method turns off an appliance by:
        1. Validating the appliance_id parameter
        2. Checking if the appliance exists in the database
        3. Verifying the appliance is connected to the network
        4. Verifying the appliance is currently in 'on' state
        5. Updating the power_state to 'off' and recording the timestamp
        6. Returning success status and timestamp

        Args:
            appliance_id: Unique identifier of the appliance to turn off

        Returns:
            dict: Contains 'success' (bool) and 'timestamp' (str in yyyy-mm-dd HH:MM:SS format)

        Raises:
            ValueError: If appliance_id is empty or invalid format
            RuntimeError: If appliance not found, not connected, or not in 'on' state
        """
        from datetime import datetime

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

        # Access the database
        db = self.db

        # Get the appliance table from database
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if the appliance exists in the database
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with id '{appliance_id}' not found")

        # Retrieve the appliance record
        appliance = appliance_table[appliance_id]

        # Verify pre-condition: appliance must be connected to network
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        # Verify pre-condition: appliance must be in 'on' state
        if appliance.power_state != 'on':
            raise RuntimeError(f"Appliance '{appliance_id}' is not in 'on' state (current state: {appliance.power_state})")

        # Record the current timestamp when turning off the appliance
        current_time = datetime.now()

        # Update the appliance power state to 'off'
        appliance.power_state = 'off'
        appliance.power_state_updated_at = current_time

        # Save the updated appliance back to the database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

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

        # Return success response with timestamp
        return {
            'success': True,
            'timestamp': timestamp_str
        }

    @is_tool()
    def lock_appliance(self, appliance_id: str) -> dict:
        """
        Lock an appliance to prevent physical button operations, typically for child safety.

        This method locks the specified appliance by setting its lock_status to 'locked'.
        The appliance must support lock functionality (have lock_status field).

        Args:
            appliance_id: Unique identifier of the appliance to lock

        Returns:
            dict: Contains success status and current lock status
                - success (bool): Whether the appliance was locked successfully
                - lock_status (str): Current lock status after operation

        Raises:
            RuntimeError: If appliance not found, not connected, or doesn't support locking
        """
        from datetime import datetime

        # Input validation: check appliance_id is not empty
        if not appliance_id or not isinstance(appliance_id, str):
            raise RuntimeError("Invalid appliance_id: must be a non-empty string")

        # Access the database
        db = self.db

        # Retrieve the appliance table
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if the appliance exists
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with id '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Pre-condition check: Appliance must be connected
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        # Pre-condition check: Appliance must support lock functionality
        # Check if lock_status field exists and is not None (supports locking)
        if not hasattr(appliance, 'lock_status'):
            raise RuntimeError(f"Appliance '{appliance_id}' does not support lock functionality")

        # Check if already locked
        current_lock_status = appliance.lock_status
        if current_lock_status == 'locked':
            # Already locked, return success with current status
            return {
                'success': True,
                'lock_status': 'locked'
            }

        # Lock the appliance by updating lock_status
        appliance.lock_status = 'locked'

        # Update the appliance in the database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Return success result
        return {
            'success': True,
            'lock_status': 'locked'
        }

    @is_tool()
    def get_appliance_remaining_time(self, appliance_id: str):
        """
        Get the estimated remaining time for the current operation of an appliance.

        This method retrieves the remaining operation time based on the appliance's 
        current progress and calculates the estimated completion time.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Contains remaining_minutes (int) and completion_time (str in yyyy-mm-dd HH:MM:SS format)

        Raises:
            KeyError: If appliance_id does not exist in the database
            ValueError: If appliance is not in running state or progress information is unavailable
        """
        from datetime import datetime, timedelta

        # Get database instance
        db = self.db

        # Retrieve appliance table data
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if appliance exists
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with id '{appliance_id}' not found")

        # Get appliance instance
        appliance = appliance_table[appliance_id]

        # Verify appliance is in running state (pre-condition check)
        if appliance.operation_status != 'running':
            raise ValueError(f"Appliance '{appliance_id}' is not in running state. Current status: {appliance.operation_status}")

        # Check if progress information is available
        if appliance.progress_percentage is None:
            raise ValueError(f"Progress information not available for appliance '{appliance_id}'")

        # Validate progress percentage is within valid range
        if appliance.progress_percentage < 0 or appliance.progress_percentage > 100:
            raise ValueError(f"Invalid progress percentage: {appliance.progress_percentage}")

        # Get current time
        current_time = datetime.now()

        # Calculate remaining time based on progress
        # If progress is 100%, operation is complete (0 minutes remaining)
        if appliance.progress_percentage >= 100:
            remaining_minutes = 0
        else:
            # Get power_state_updated_at as the start time of current operation
            # This represents when the appliance last changed state (likely when it started running)
            operation_start_time = appliance.power_state_updated_at

            # Calculate elapsed time since operation started
            elapsed_time = current_time - operation_start_time
            elapsed_minutes = elapsed_time.total_seconds() / 60

            # Avoid division by zero
            if appliance.progress_percentage == 0:
                # If no progress yet but running, estimate based on typical operation time
                # Default to 60 minutes as a reasonable estimate
                remaining_minutes = 60
            else:
                # Calculate total estimated time based on current progress rate
                # Formula: total_time = elapsed_time / (progress / 100)
                total_estimated_minutes = elapsed_minutes / (appliance.progress_percentage / 100.0)

                # Calculate remaining time
                remaining_minutes = int(total_estimated_minutes - elapsed_minutes)

                # Ensure remaining minutes is non-negative
                if remaining_minutes < 0:
                    remaining_minutes = 0

        # Calculate completion time
        completion_time = current_time + timedelta(minutes=remaining_minutes)

        # Format completion time as yyyy-mm-dd HH:MM:SS
        completion_time_str = completion_time.strftime("%Y-%m-%d %H:%M:%S")

        # Return result
        return {
            'remaining_minutes': remaining_minutes,
            'completion_time': completion_time_str
        }

    @is_tool()
    def get_appliance_notification_settings(self, appliance_id: str):
        """
        Retrieve the current notification settings for an appliance.

        This method fetches the notification configuration for a specific appliance,
        including whether notifications are enabled and what types of notifications
        are configured.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing:
                - notifications_enabled (bool): Whether notifications are enabled
                - enabled_notification_types (list): List of enabled notification types

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

        # Retrieve the appliance_notification_setting table
        notification_settings_table = getattr(db, 'appliance_notification_setting', None)

        # Check if the table exists
        if notification_settings_table is None:
            raise RuntimeError("Appliance notification settings table not found in database")

        # Check if the appliance_id exists in the notification settings table
        if appliance_id not in notification_settings_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found in notification settings")

        # Retrieve the notification setting record for this appliance
        setting_record = notification_settings_table[appliance_id]

        # Extract the notifications_enabled flag
        notifications_enabled = setting_record.notifications_enabled

        # Parse the enabled_notification_types from comma-separated string to list
        # Handle the case where enabled_notification_types might be None or empty
        enabled_notification_types = []
        if setting_record.enabled_notification_types:
            # Split by comma and strip whitespace from each type
            enabled_notification_types = [
                notification_type.strip() 
                for notification_type in setting_record.enabled_notification_types.split(',')
                if notification_type.strip()  # Filter out empty strings
            ]

        # Construct and return the result dictionary
        return {
            'notifications_enabled': notifications_enabled,
            'enabled_notification_types': enabled_notification_types
        }

    @is_tool()
    def get_appliance_mute_status(self, appliance_id: str) -> dict:
        """
        Retrieve the current mute status of an appliance

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing the mute status

        Raises:
            KeyError: If the appliance is not found
        """
        # Access the database instance
        db = self.db

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

        # Retrieve the appliance table from database
        appliance_table = getattr(db, 'appliance', None)

        # Check if appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if the appliance exists in the table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with id '{appliance_id}' not found")

        # Retrieve the appliance object
        appliance = appliance_table[appliance_id]

        # Get the mute status from the appliance
        # The mute_status field is optional in the schema, so we need to handle None case
        mute_status = appliance.mute_status

        # If mute_status is None, it means the appliance doesn't support sound control
        # or the status hasn't been set yet
        if mute_status is None:
            raise ValueError(f"Appliance '{appliance_id}' does not have mute status information (may not support sound control)")

        # Return the mute status in the required format
        return {
            'mute_status': mute_status
        }

    @is_tool()
    def get_appliance_schedules(self, appliance_id: str):
        """
        Retrieve all scheduled operations for a specific appliance

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing list of scheduled operations

        Raises:
            KeyError: If appliance_id is not found in the schedule database
        """
        # Get database instance
        db = self.db

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

        # Get the appliance_schedule table from database
        appliance_schedule_table = getattr(db, 'appliance_schedule', None)

        # Check if the table exists
        if appliance_schedule_table is None:
            raise KeyError(f"appliance_schedule table not found in database")

        # Initialize list to store matching schedules
        matching_schedules = []

        # Iterate through all schedules in the table
        # appliance_schedule_table is a Dict[str, ApplianceSchedule] where key is schedule_id
        for schedule_id, schedule in appliance_schedule_table.items():
            # Check if this schedule belongs to the requested appliance
            if schedule.appliance_id == appliance_id:
                # Convert datetime objects to string format "yyyy-mm-dd HH:MM:SS"
                scheduled_time_str = schedule.scheduled_time.strftime("%Y-%m-%d %H:%M:%S")

                # Build schedule dictionary with required fields
                schedule_dict = {
                    'schedule_id': schedule.schedule_id,
                    'scheduled_time': scheduled_time_str,
                    'action': schedule.command  # Map 'command' field to 'action' in output
                }

                # Add to results list
                matching_schedules.append(schedule_dict)

        # If no schedules found for this appliance_id, raise KeyError
        if len(matching_schedules) == 0:
            raise KeyError(f"No schedules found for appliance_id: {appliance_id}")

        # Return the schedules in the expected format
        return {
            'schedules': matching_schedules
        }

    @is_tool()
    def get_appliance_eco_mode_status(self, appliance_id: str):
        """
        Retrieve the current eco mode status of an appliance.

        This method queries the appliance database to get the eco_mode_enabled status
        for the specified appliance.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: A dictionary containing:
                - eco_mode_enabled (bool): Whether eco mode is enabled for the appliance

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

        # Validate that appliance_id is provided and not empty
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

        # Get the appliance table from the database
        appliance_table = getattr(db, 'appliance', None)

        # Check if the appliance table exists
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if the appliance exists in the table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with ID '{appliance_id}' not found")

        # Retrieve the appliance object
        appliance = appliance_table[appliance_id]

        # Extract the eco_mode_enabled status with explicit boolean conversion
        eco_mode_enabled = bool(appliance.eco_mode_enabled)

        # Return the result in the specified format
        return {
            'eco_mode_enabled': eco_mode_enabled
        }

    @is_tool()
    def get_appliance_operation_status(self, appliance_id: str) -> dict:
        """
        Retrieve the current operation status of an appliance.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Dictionary containing operation_status and progress_percentage

        Raises:
            KeyError: If appliance_id is not found in the database
        """
        # Input parameter validation
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

        # Access the database
        db = self.db

        # Get the appliance table from the database
        appliance_table = getattr(db, 'appliance', None)

        # Check if the appliance table exists
        if appliance_table is None:
            raise KeyError(f"Appliance table not found in database")

        # Check if the appliance_id exists in the appliance table
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with appliance_id '{appliance_id}' not found")

        # Retrieve the appliance object
        appliance = appliance_table[appliance_id]

        # Extract operation status and progress percentage
        operation_status = appliance.operation_status
        progress_percentage = appliance.progress_percentage

        # Construct the return dictionary
        # Note: progress_percentage can be None (optional field), which is valid
        result = {
            'operation_status': operation_status,
            'progress_percentage': progress_percentage
        }

        return result

    @is_tool()
    def disable_appliance_notifications(self, appliance_id: str) -> dict:
        """
        Disable push notifications for an appliance

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Contains 'success' boolean indicating whether notifications were disabled successfully

        Raises:
            RuntimeError: If appliance notification settings not found or notifications already disabled
        """
        from datetime import datetime

        # Access the database
        db = self.db

        # Retrieve the appliance_notification_setting table
        notification_settings = getattr(db, 'appliance_notification_setting', None)

        # Check if the table exists
        if notification_settings is None:
            raise RuntimeError("Appliance notification settings table not found")

        # Check if the appliance_id exists in the notification settings
        if appliance_id not in notification_settings:
            raise RuntimeError(f"Notification settings not found for appliance_id: {appliance_id}")

        # Retrieve the notification setting for the appliance
        setting = notification_settings[appliance_id]

        # Check pre-condition: notifications must be currently enabled
        if not setting.notifications_enabled:
            raise RuntimeError(f"Notifications are already disabled for appliance_id: {appliance_id}")

        # Disable notifications by setting notifications_enabled to False
        setting.notifications_enabled = False

        # Update the timestamp to reflect when the setting was changed in "yyyy-mm-dd HH:MM:SS" format
        setting.updated_at = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Update the notification settings in the database
        notification_settings[appliance_id] = setting
        setattr(db, 'appliance_notification_setting', notification_settings)

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

    @is_tool()
    def cancel_appliance_schedule(self, schedule_id: str) -> dict:
        """
        Cancel a scheduled operation for an appliance.

        This method cancels an existing appliance schedule by updating its status to 'cancelled'.
        It validates that the schedule exists before attempting to cancel it.

        Args:
            schedule_id: Unique identifier of the schedule to cancel

        Returns:
            dict: A dictionary containing:
                - success (bool): Whether the schedule was cancelled successfully

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

        # Get the appliance_schedule table
        appliance_schedule_table = getattr(db, 'appliance_schedule', None)

        # Check if the table exists
        if appliance_schedule_table is None:
            raise KeyError(f"Schedule with ID '{schedule_id}' not found: appliance_schedule table does not exist")

        # Check if the schedule exists in the table
        if schedule_id not in appliance_schedule_table:
            raise KeyError(f"Schedule with ID '{schedule_id}' not found")

        # Retrieve the schedule item
        schedule_item = appliance_schedule_table[schedule_id]

        # Update the schedule status to 'cancelled'
        schedule_item.status = 'cancelled'

        # Write the updated schedule back to the database
        appliance_schedule_table[schedule_id] = schedule_item
        setattr(db, 'appliance_schedule', appliance_schedule_table)

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

    @is_tool()
    def set_appliance_humidity_level(self, appliance_id: str, humidity_percentage: int):
        """
        Set the target humidity level for appliances like humidifiers or dehumidifiers.

        This method updates the target_humidity field for a specified appliance and returns
        the success status along with the current humidity reading.

        Args:
            appliance_id: Unique identifier of the appliance
            humidity_percentage: Target humidity level as percentage (0-100)

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

        Raises:
            ValueError: If appliance_id is invalid, appliance doesn't exist, 
                       humidity_percentage is out of range (0-100), or appliance 
                       doesn't support humidity control
        """
        # Validate input parameters
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

        if not isinstance(humidity_percentage, int):
            raise ValueError("humidity_percentage must be an integer")

        # Validate humidity percentage range (0-100)
        if humidity_percentage < 0 or humidity_percentage > 100:
            raise ValueError("humidity_percentage must be between 0 and 100")

        # Access the database
        db = self.db

        # Get the appliance table
        appliance_table = getattr(db, "appliance", None)
        if appliance_table is None:
            raise ValueError("Appliance table not found in database")

        # Check if appliance exists
        if appliance_id not in appliance_table:
            raise ValueError(f"Appliance with id '{appliance_id}' does not exist")

        # Retrieve the appliance record
        appliance = appliance_table[appliance_id]

        # Check if appliance supports humidity control
        # An appliance supports humidity control if it has target_humidity field capability
        # We verify this by checking if the appliance type typically supports humidity
        # or if the current_humidity/target_humidity fields are available (not None initially set)
        # For this implementation, we check if the appliance has humidity-related fields defined

        # Common appliance types that support humidity control
        humidity_control_types = [
            "humidifier", "dehumidifier", "air_conditioner", 
            "air_purifier", "hvac", "climate_control"
        ]

        # Check if appliance type supports humidity control
        appliance_type_lower = appliance.appliance_type.lower()
        supports_humidity = any(hc_type in appliance_type_lower for hc_type in humidity_control_types)

        if not supports_humidity:
            raise ValueError(
                f"Appliance type '{appliance.appliance_type}' does not support humidity control. "
                f"Only humidifiers, dehumidifiers, and similar appliances support this feature."
            )

        # Update the target humidity level
        appliance.target_humidity = humidity_percentage

        # Get current humidity reading (may be None if sensor hasn't provided reading yet)
        current_humidity = appliance.current_humidity

        # If current_humidity is None, initialize it to a reasonable default
        # In real scenarios, this would come from the actual sensor
        if current_humidity is None:
            current_humidity = 50  # Default middle value
            appliance.current_humidity = current_humidity

        # Save the updated appliance back to database
        appliance_table[appliance_id] = appliance
        setattr(db, "appliance", appliance_table)

        # Return success response with current humidity
        return {
            "success": True,
            "current_humidity": current_humidity
        }

    @is_tool()
    def turn_on_appliance(self, appliance_id: str) -> dict:
        """
        Turn on a specific smart home appliance remotely

        Args:
            appliance_id: Unique identifier of the appliance to turn on

        Returns:
            dict: Contains success status and timestamp of operation

        Raises:
            RuntimeError: If appliance not found, not connected, or already in 'on' state
        """
        from datetime import datetime

        # Input validation - check if appliance_id is provided and valid
        if not appliance_id or not isinstance(appliance_id, str):
            raise RuntimeError("Invalid appliance_id: must be a non-empty string")

        # Access the database
        db = self.db

        # Retrieve the appliance table from database
        appliance_table = getattr(db, 'appliance', None)
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if the appliance exists in the database
        if appliance_id not in appliance_table:
            raise RuntimeError(f"Appliance with ID '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Pre-condition check: Appliance must be connected to network
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network")

        # Pre-condition check: Appliance must be in off or standby state
        # (not already on)
        current_power_state = appliance.power_state.lower()
        if current_power_state not in ['off', 'standby']:
            raise RuntimeError(
                f"Appliance '{appliance_id}' is already in '{appliance.power_state}' state. "
                f"Can only turn on appliances in 'off' or 'standby' state"
            )

        # Record the current timestamp for the operation
        operation_timestamp = datetime.now()

        # Update the appliance power state to 'on'
        appliance.power_state = 'on'
        appliance.power_state_updated_at = operation_timestamp

        # Update the appliance's last_seen timestamp to current time
        appliance.last_seen = operation_timestamp

        # Save the updated appliance back to the database
        appliance_table[appliance_id] = appliance
        setattr(db, 'appliance', appliance_table)

        # Format the timestamp as required: "yyyy-mm-dd HH:MM:SS"
        timestamp_str = operation_timestamp.strftime("%Y-%m-%d %H:%M:%S")

        # Return success response with timestamp
        return {
            'success': True,
            'timestamp': timestamp_str
        }

    @is_tool()
    def get_appliance_signal_strength(self, appliance_id: str) -> dict:
        """
        Retrieve the WiFi or network signal strength of an appliance.

        This method queries the appliance database to retrieve the signal strength
        information for a specific appliance identified by its appliance_id.

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: A dictionary containing:
                - signal_strength_dbm: Signal strength in dBm (integer)
                - signal_quality: Signal quality rating (string: excellent, good, fair, poor)

        Raises:
            KeyError: If the appliance_id does not exist in the database
            ValueError: If appliance_id is invalid or signal information is unavailable
        """
        # Validate input parameter
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

        # Access the database
        db = self.db

        # Get the appliance table from the database
        appliance_table = getattr(db, 'appliance', None)

        # Check if the appliance table exists
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Retrieve the appliance by appliance_id
        # The table is structured as Dict[str, Appliance] where key is appliance_id
        if appliance_id not in appliance_table:
            raise KeyError(f"Appliance with id '{appliance_id}' not found")

        # Get the appliance object
        appliance = appliance_table[appliance_id]

        # Extract signal strength information
        # Note: signal_strength_dbm and signal_quality are optional fields
        signal_strength_dbm = appliance.signal_strength_dbm
        signal_quality = appliance.signal_quality

        # Validate that signal information is available
        # If the appliance is not connected or doesn't have signal info, these might be None
        if signal_strength_dbm is None or signal_quality is None:
            raise KeyError(f"Signal strength information not available for appliance '{appliance_id}'")

        # Return the signal strength information as a dictionary
        return {
            'signal_strength_dbm': signal_strength_dbm,
            'signal_quality': signal_quality
        }

    @is_tool()
    def create_appliance_operation_sequence(self, appliance_id: str, operations: list) -> dict:
        """
        Create a sequence of operations to be executed on an appliance in order.

        Args:
            appliance_id: Unique identifier of the appliance
            operations: List of operations to execute in sequence, each operation is a dict
                       containing 'command', optional 'parameters', and optional 'delay_seconds'

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

        Raises:
            ValueError: If appliance_id is empty, operations list is empty or invalid
        """
        import json
        import secrets
        import hashlib
        from datetime import datetime
        
        # Validate appliance_id
        if not appliance_id or not isinstance(appliance_id, str):
            raise ValueError("appliance_id must be a non-empty string")

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

        if len(operations) == 0:
            raise ValueError("operations list cannot be empty")

        # Validate each operation in the list
        for idx, operation in enumerate(operations):
            if not isinstance(operation, dict):
                raise ValueError(f"Operation at index {idx} must be a dictionary")

            # Each operation must have a 'command' field
            if 'command' not in operation:
                raise ValueError(f"Operation at index {idx} must contain 'command' field")

            if not isinstance(operation['command'], str) or not operation['command']:
                raise ValueError(f"Operation at index {idx} 'command' must be a non-empty string")

            # Validate optional 'delay_seconds' field if present
            if 'delay_seconds' in operation:
                if not isinstance(operation['delay_seconds'], (int, float)):
                    raise ValueError(f"Operation at index {idx} 'delay_seconds' must be a number")
                if operation['delay_seconds'] < 0:
                    raise ValueError(f"Operation at index {idx} 'delay_seconds' cannot be negative")

            # Validate optional 'parameters' field if present
            if 'parameters' in operation:
                if not isinstance(operation['parameters'], dict):
                    raise ValueError(f"Operation at index {idx} 'parameters' must be a dictionary")

        # Access the database
        db = self.db

        # Get operation_sequence table, initialize if not exists
        operation_sequence_table = getattr(db, 'operation_sequence', None)
        if operation_sequence_table is None:
            operation_sequence_table = {}
            setattr(db, 'operation_sequence', operation_sequence_table)

        # Generate unique sequence_id (VARCHAR(10) max length)
        max_attempts = 100
        sequence_id = None

        for _ in range(max_attempts):
            # Generate 10-character ID to fit VARCHAR(10)
            random_suffix = hashlib.sha256(secrets.token_bytes(32)).hexdigest()[:10]
            sequence_id = random_suffix

            if sequence_id not in operation_sequence_table:
                break
        else:
            raise RuntimeError("Failed to generate unique sequence_id after maximum attempts")

        # Convert operations list to JSON string for storage
        operations_json = json.dumps(operations)

        # Get current timestamp in required format "yyyy-mm-dd HH:MM:SS"
        current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S")

        # Create new OperationSequence instance
        new_sequence = OperationSequence(
            sequence_id=sequence_id,
            appliance_id=appliance_id,
            operations=operations_json,
            status="pending",  # Initial status is pending
            created_at=current_time,
            executed_at=None  # Not executed yet
        )

        # Store the new sequence in the database
        operation_sequence_table[sequence_id] = new_sequence
        setattr(db, 'operation_sequence', operation_sequence_table)

        # Return success response
        return {
            'sequence_id': sequence_id,
            'success': True
        }

    @is_tool()
    def schedule_appliance_operation(self, appliance_id: str, scheduled_time: str, command: str) -> dict:
        """
        Schedule an appliance to turn on or off at a specific time.

        Args:
            appliance_id: Unique identifier of the appliance
            scheduled_time: Time to perform the action in yyyy-mm-dd HH:MM:SS format
            command: Action to perform (turn_on, turn_off)

        Returns:
            dict: Contains success status and schedule_id

        Raises:
            ValueError: If parameters are invalid or scheduling fails
        """
        import secrets
        import hashlib
        from datetime import datetime

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

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

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

        # Validate command is one of the supported actions
        valid_commands = ['turn_on', 'turn_off']
        if command not in valid_commands:
            raise ValueError(f"command must be one of {valid_commands}, got '{command}'")

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

        # Validate scheduled time is in the future
        current_time = datetime.now()
        if scheduled_datetime <= current_time:
            raise ValueError(f"scheduled_time must be in the future, got '{scheduled_time}' which is not after current time")

        # Access the database
        db = self.db

        # Get or initialize the appliance_schedule table
        appliance_schedule_table = getattr(db, 'appliance_schedule', None)
        if appliance_schedule_table is None:
            appliance_schedule_table = {}
            setattr(db, 'appliance_schedule', appliance_schedule_table)

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

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

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

        # Create new schedule entry with status 'pending'
        new_schedule = ApplianceSchedule(
            schedule_id=schedule_id,
            appliance_id=appliance_id,
            scheduled_time=scheduled_datetime,
            command=command,
            status='pending',
            created_at=created_at
        )

        # Add the new schedule to the database
        appliance_schedule_table[schedule_id] = new_schedule
        setattr(db, 'appliance_schedule', appliance_schedule_table)

        # Return success response with the generated schedule_id
        return {
            'success': True,
            'schedule_id': schedule_id
        }

    @is_tool()
    def start_appliance_firmware_update(self, appliance_id: str) -> dict:
        """
        Initiate a firmware update for an appliance.

        This method starts the firmware update process for a specified appliance.
        It validates that the appliance exists, is connected, and has a firmware 
        update available (indicated by checking if current firmware is not the latest).

        Args:
            appliance_id: Unique identifier of the appliance

        Returns:
            dict: Contains 'success' (bool) indicating if update was initiated,
                  and 'estimated_duration_minutes' (int) for estimated update duration

        Raises:
            RuntimeError: If appliance not found, not connected, or update not available
        """
        from datetime import datetime

        # Input validation - check appliance_id is provided and not empty
        if not appliance_id or not isinstance(appliance_id, str):
            raise RuntimeError("Invalid appliance_id: must be a non-empty string")

        # Access the database
        db = self.db

        # Retrieve appliance table from database
        appliance_table = getattr(db, "appliance", None)
        if appliance_table is None:
            raise RuntimeError("Appliance table not found in database")

        # Check if appliance exists in the database
        appliance = appliance_table.get(appliance_id)
        if appliance is None:
            raise RuntimeError(f"Appliance with id '{appliance_id}' not found")

        # Pre-condition check: Appliance must be connected
        if not appliance.connected:
            raise RuntimeError(f"Appliance '{appliance_id}' is not connected to network. Cannot initiate firmware update.")

        # Pre-condition check: Firmware update must be available
        # Since there's no explicit "update_available" field in the schema,
        # we assume that calling this method implies an update is available
        # The pre-condition states "Firmware update must be available" - we trust the caller has verified this

        # Calculate estimated duration based on appliance type
        # Different appliance types may have different firmware sizes and update times
        appliance_type = appliance.appliance_type.lower()

        # Estimate duration based on appliance complexity
        # Simpler devices = shorter updates, complex devices = longer updates
        duration_map = {
            'air_conditioner': 20,
            'washing_machine': 15,
            'refrigerator': 18,
            'dishwasher': 15,
            'oven': 12,
            'microwave': 10,
            'television': 15,
            'smart_speaker': 8,
            'thermostat': 10,
            'security_camera': 12,
            'door_lock': 10,
            'light': 5,
            'fan': 8,
            'vacuum': 15,
            'water_heater': 12
        }

        # Default duration if appliance type not in map
        estimated_duration = 15

        # Check if appliance_type matches any known type (using fuzzy matching for robustness)
        if appliance_type in duration_map:
            estimated_duration = duration_map[appliance_type]
        else:
            # Try fuzzy matching to find closest appliance type
            from thefuzz import process
            match_result = process.extractOne(appliance_type, list(duration_map.keys()))
            if match_result and match_result[1] >= 70:  # 70% similarity threshold
                matched_type = match_result[0]
                estimated_duration = duration_map[matched_type]

        # Update appliance status to indicate firmware update is in progress
        # Set operation_status to 'running' to indicate update process has started
        appliance.operation_status = 'running'
        appliance.progress_percentage = 0  # Reset progress to 0 at start

        # Update the last_seen timestamp to current time
        current_time = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
        appliance.last_seen = current_time

        # Save the updated appliance back to database
        appliance_table[appliance_id] = appliance
        setattr(db, "appliance", appliance_table)

        # Return success response with estimated duration
        return {
            'success': True,
            'estimated_duration_minutes': estimated_duration
        }

    @is_tool()
    def start_appliance_timer(self, appliance_id: str, duration_minutes: int, command: str) -> dict:
        """
        Start a countdown timer for an appliance to perform an action after specified duration.

        Args:
            appliance_id: Unique identifier of the appliance
            duration_minutes: Timer duration in minutes
            command: Action to perform when timer expires (turn_off, turn_on)

        Returns:
            dict: Contains success status and timer_end_time in 'yyyy-mm-dd HH:MM:SS' format

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

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

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

        # Validate command parameter
        valid_commands = ['turn_off', 'turn_on']
        if not isinstance(command, str) or command not in valid_commands:
            raise ValueError(f"command must be one of {valid_commands}")

        # Access the database
        db = self.db

        # Get the appliance_timer table
        appliance_timer_table = getattr(db, 'appliance_timer', None)
        if appliance_timer_table is None:
            raise ValueError("appliance_timer table not found in database")

        # Calculate timer end time
        current_time = datetime.now()
        timer_end_time = current_time + timedelta(minutes=duration_minutes)

        # Create or update the timer record for this appliance
        # Check if timer already exists for this appliance
        existing_timer = appliance_timer_table.get(appliance_id)

        if existing_timer:
            # Update existing timer
            existing_timer.timer_active = True
            existing_timer.duration_minutes = duration_minutes
            existing_timer.remaining_minutes = duration_minutes
            existing_timer.timer_action = command
            existing_timer.timer_end_time = timer_end_time
            existing_timer.started_at = current_time
        else:
            # Create new timer record
            new_timer = ApplianceTimer(
                appliance_id=appliance_id,
                timer_active=True,
                duration_minutes=duration_minutes,
                remaining_minutes=duration_minutes,
                timer_action=command,
                timer_end_time=timer_end_time,
                started_at=current_time
            )
            # Add new timer to the table
            appliance_timer_table[appliance_id] = new_timer

        # Update the database table
        setattr(db, 'appliance_timer', appliance_timer_table)

        # Format timer_end_time as string in 'yyyy-mm-dd HH:MM:SS' format
        timer_end_time_str = timer_end_time.strftime('%Y-%m-%d %H:%M:%S')

        # Return success response with timer end time
        return {
            'success': True,
            'timer_end_time': timer_end_time_str
        }
