"""
Database schemas for smart_home_Appliance_Remote_Operation

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

Generated by ClassDBAgent.
"""

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

@with_instance_key("appliance_id")
class Appliance(BaseModel, ThreadSafeBase["Appliance"]):
    appliance_id: str = Field(default=..., description="Unique identifier of the appliance")
    appliance_type: str = Field(default=..., description="Type of the appliance (air_conditioner, washing_machine, etc.)")
    power_state: str = Field(default=..., description="Current power state (on, off, standby)")
    power_state_updated_at: datetime = Field(default=..., description="Last time the power state was updated")
    target_temperature: Optional[float] = Field(default=None, description="Target temperature setting in Celsius")
    current_temperature: Optional[float] = Field(default=None, description="Current temperature reading in Celsius")
    mode: Optional[str] = Field(default=None, description="Current operating mode (cooling, heating, fan, auto)")
    fan_speed: Optional[str] = Field(default=None, description="Current fan speed setting (low, medium, high, auto)")
    lock_status: Optional[str] = Field(default=None, description="Current lock status (locked, unlocked)")
    brightness_percentage: Optional[int] = Field(default=None, description="Current brightness level percentage (0-100)")
    mute_status: Optional[str] = Field(default=None, description="Current mute status (muted, unmuted)")
    volume_percentage: Optional[int] = Field(default=None, description="Current volume level percentage (0-100)")
    operation_status: str = Field(default=..., description="Current operation status (idle, running, paused, stopped, completed)")
    progress_percentage: Optional[int] = Field(default=None, description="Progress of current operation as percentage")
    eco_mode_enabled: bool = Field(default=..., description="Whether eco mode is enabled")
    swing_mode: Optional[str] = Field(default=None, description="Current swing mode setting (horizontal, vertical, both, off)")
    target_humidity: Optional[int] = Field(default=None, description="Target humidity setting percentage (0-100)")
    current_humidity: Optional[int] = Field(default=None, description="Current humidity reading percentage (0-100)")
    connected: bool = Field(default=..., description="Whether the appliance is connected to network")
    last_seen: datetime = Field(default=..., description="Last time the appliance was seen online")
    firmware_version: str = Field(default=..., description="Current firmware version")
    firmware_updated_at: datetime = Field(default=..., description="Date of last firmware update")
    signal_strength_dbm: Optional[int] = Field(default=None, description="Network signal strength in dBm")
    signal_quality: Optional[str] = Field(default=None, description="Signal quality rating (excellent, good, fair, poor)")

@with_instance_key("appliance_id")
class ApplianceTimer(BaseModel, ThreadSafeBase["ApplianceTimer"]):
    appliance_id: str = Field(
        default=...,
        description="Unique identifier of the appliance"
    )
    timer_active: bool = Field(
        default=...,
        description="Whether a timer is currently active"
    )
    duration_minutes: Optional[int] = Field(
        default=None,
        description="Timer duration in minutes"
    )
    remaining_minutes: Optional[int] = Field(
        default=None,
        description="Remaining time in minutes"
    )
    timer_action: Optional[str] = Field(
        default=None,
        description="Action to be performed when timer expires (turn_off, turn_on)"
    )
    timer_end_time: Optional[datetime] = Field(
        default=None,
        description="Time when the timer will expire"
    )
    started_at: Optional[datetime] = Field(
        default=None,
        description="Time when the timer was started"
    )

@with_instance_key("schedule_id")
class ApplianceSchedule(BaseModel, ThreadSafeBase["ApplianceSchedule"]):
    schedule_id: str = Field(default=..., description="Unique identifier of the schedule")
    appliance_id: str = Field(default=..., description="Unique identifier of the appliance")
    scheduled_time: datetime = Field(default=..., description="Time to perform the action")
    command: str = Field(default=..., description="Action to perform (turn_on, turn_off)")
    status: str = Field(default=..., description="Schedule status (pending, executed, cancelled)")
    created_at: datetime = Field(default=..., description="Time when the schedule was created")

@with_instance_key("appliance_id")
class ApplianceEnergyConsumption(BaseModel, ThreadSafeBase["ApplianceEnergyConsumption"]):
    appliance_id: str = Field(default=..., description="Unique identifier of the appliance")
    timestamp: datetime = Field(default=..., description="Time of the reading")
    current_power_watts: Decimal = Field(default=..., description="Current power consumption in watts")
    total_energy_kwh: Decimal = Field(default=..., description="Cumulative energy consumed in kilowatt-hours")

    @validator("timestamp", pre=True)
    def strip_timestamp(cls, v):
        # Remove leading/trailing spaces before parsing
        if isinstance(v, str):
            return v.strip()
        return v

@with_instance_key("appliance_id")
class ApplianceError(BaseModel, ThreadSafeBase["ApplianceError"]):
    appliance_id: str = Field(default=..., description="Unique identifier of the appliance")
    has_error: bool = Field(default=..., description="Whether the appliance has an error")
    error_code: Optional[str] = Field(default=None, description="Error code if any")
    error_message: Optional[str] = Field(default=None, description="Human-readable error message")
    error_occurred_at: Optional[datetime] = Field(default=None, description="Time when the error occurred")
    error_cleared_at: Optional[datetime] = Field(default=None, description="Time when the error was cleared")

@with_instance_key("appliance_id")
class ApplianceNotificationSetting(BaseModel, ThreadSafeBase["ApplianceNotificationSetting"]):      
    appliance_id: str = Field(default=..., description="Unique identifier of the appliance")
    notifications_enabled: bool = Field(default=..., description="Whether notifications are enabled")
    enabled_notification_types: Optional[str] = Field(default=None, description="Comma-separated list of enabled notification types")
    updated_at: datetime = Field(default=..., description="Time when settings were last updated")

@with_instance_key("appliance_id")
class ApplianceUsageStatistic(BaseModel, ThreadSafeBase["ApplianceUsageStatistic"]):
    appliance_id: str = Field(..., description="Unique identifier of the appliance")
    start_date: date = Field(..., description="Start date of the statistics period")
    end_date: date = Field(..., description="End date of the statistics period")
    total_runtime_hours: float = Field(..., description="Total runtime in hours")
    operation_count: int = Field(..., description="Number of times the appliance was operated")
    total_energy_kwh: float = Field(..., description="Total energy consumed in kilowatt-hours")

    @field_validator("start_date", "end_date", mode="before")
    @classmethod
    def parse_date_from_datetime_str(cls, value):
        # Accept both "YYYY-MM-DD" and "YYYY-MM-DD HH:MM:SS" formats
        if isinstance(value, date):
            return value
        if isinstance(value, str):
            try:
                # Try parsing as date
                return datetime.strptime(value, "%Y-%m-%d").date()
            except ValueError:
                # Try parsing as datetime
                try:
                    dt = datetime.strptime(value, "%Y-%m-%d %H:%M:%S")
                    return dt.date()
                except ValueError:
                    raise ValueError(
                        f"Invalid date format for field, expected 'YYYY-MM-DD' or 'YYYY-MM-DD HH:MM:SS', got '{value}'"
                    )
        raise ValueError("Unsupported type for date field")

@with_instance_key("appliance_id")
class ApplianceFeature(BaseModel, ThreadSafeBase["ApplianceFeature"]):
    appliance_id: str = Field(
        default=..., 
        description="Unique identifier of the appliance"
    )
    features: str = Field(
        default=..., 
        description="Comma-separated list of supported features"
    )
    available_modes: Optional[str] = Field(
        default=None, 
        description="Comma-separated list of available operating modes"
    )

@with_instance_key("sequence_id")
class OperationSequence(BaseModel, ThreadSafeBase["OperationSequence"]):
    sequence_id: str = Field(default=..., description="Unique identifier of the sequence")
    appliance_id: str = Field(default=..., description="Unique identifier of the appliance")
    operations: str = Field(default=..., description="JSON array of operations to execute in sequence")
    status: str = Field(default=..., description="Sequence status (pending, executing, completed, failed)")
    created_at: datetime = Field(default=..., description="Time when the sequence was created")
    executed_at: Optional[datetime] = Field(default=None, description="Time when the sequence was executed")  # executed_at is optional

class SmartHomeDB(DB):
    """Database containing all smart_home_Appliance_Remote_Operation-related data"""
    appliance: Optional[Dict[str, Appliance]] = Field(
        default=None,
        description="Schema Appliance"
    )
    appliance_timer: Optional[Dict[str, ApplianceTimer]] = Field(
        default=None,
        description="Schema ApplianceTimer"
    )
    appliance_schedule: Optional[Dict[str, ApplianceSchedule]] = Field(
        default=None,
        description="Schema ApplianceSchedule"
    )
    appliance_energy_consumption: Optional[Dict[str, ApplianceEnergyConsumption]] = Field(
        default=None,
        description="Schema ApplianceEnergyConsumption"
    )
    appliance_error: Optional[Dict[str, ApplianceError]] = Field(
        default=None,
        description="Schema ApplianceError"
    )
    appliance_notification_setting: Optional[Dict[str, ApplianceNotificationSetting]] = Field(
        default=None,
        description="Schema ApplianceNotificationSetting"
    )
    appliance_usage_statistic: Optional[Dict[str, ApplianceUsageStatistic]] = Field(
        default=None,
        description="Schema ApplianceUsageStatistic"
    )
    appliance_feature: Optional[Dict[str, ApplianceFeature]] = Field(
        default=None,
        description="Schema ApplianceFeature"
    )
    operation_sequence: Optional[Dict[str, OperationSequence]] = Field(
        default=None,
        description="Schema OperationSequence"
    )