smart_plug.py 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. from datetime import datetime
  2. from typing import Literal
  3. from pydantic import BaseModel, Field, model_validator
  4. class SmartPlugBase(BaseModel):
  5. name: str = Field(..., min_length=1, max_length=100)
  6. plug_type: Literal["tasmota", "homeassistant"] = "tasmota"
  7. # Tasmota fields (required when plug_type="tasmota")
  8. ip_address: str | None = Field(default=None, pattern=r"^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$")
  9. username: str | None = None
  10. password: str | None = None
  11. # Home Assistant fields (required when plug_type="homeassistant")
  12. ha_entity_id: str | None = Field(default=None, pattern=r"^(switch|light|input_boolean)\.[a-z0-9_]+$")
  13. # Home Assistant energy sensor entities (optional, for separate energy sensors)
  14. ha_power_entity: str | None = Field(default=None, pattern=r"^sensor\.[a-z0-9_]+$")
  15. ha_energy_today_entity: str | None = Field(default=None, pattern=r"^sensor\.[a-z0-9_]+$")
  16. ha_energy_total_entity: str | None = Field(default=None, pattern=r"^sensor\.[a-z0-9_]+$")
  17. printer_id: int | None = None
  18. enabled: bool = True
  19. auto_on: bool = True
  20. auto_off: bool = True
  21. off_delay_mode: Literal["time", "temperature"] = "time"
  22. off_delay_minutes: int = Field(default=5, ge=0, le=60)
  23. off_temp_threshold: int = Field(default=70, ge=30, le=150)
  24. # Power alerts
  25. power_alert_enabled: bool = False
  26. power_alert_high: float | None = Field(default=None, ge=0, le=5000) # Alert when power > this (watts)
  27. power_alert_low: float | None = Field(default=None, ge=0, le=5000) # Alert when power < this (watts)
  28. # Schedule
  29. schedule_enabled: bool = False
  30. schedule_on_time: str | None = Field(default=None, pattern=r"^([01]\d|2[0-3]):[0-5]\d$") # HH:MM format
  31. schedule_off_time: str | None = Field(default=None, pattern=r"^([01]\d|2[0-3]):[0-5]\d$") # HH:MM format
  32. # Switchbar visibility
  33. show_in_switchbar: bool = False
  34. @model_validator(mode="after")
  35. def validate_plug_type_fields(self) -> "SmartPlugBase":
  36. if self.plug_type == "tasmota" and not self.ip_address:
  37. raise ValueError("ip_address is required for Tasmota plugs")
  38. if self.plug_type == "homeassistant" and not self.ha_entity_id:
  39. raise ValueError("ha_entity_id is required for Home Assistant plugs")
  40. return self
  41. class SmartPlugCreate(SmartPlugBase):
  42. pass
  43. class SmartPlugUpdate(BaseModel):
  44. name: str | None = None
  45. plug_type: Literal["tasmota", "homeassistant"] | None = None
  46. ip_address: str | None = None
  47. ha_entity_id: str | None = None
  48. # Home Assistant energy sensor entities (optional)
  49. ha_power_entity: str | None = None
  50. ha_energy_today_entity: str | None = None
  51. ha_energy_total_entity: str | None = None
  52. printer_id: int | None = None
  53. enabled: bool | None = None
  54. auto_on: bool | None = None
  55. auto_off: bool | None = None
  56. off_delay_mode: Literal["time", "temperature"] | None = None
  57. off_delay_minutes: int | None = Field(default=None, ge=0, le=60)
  58. off_temp_threshold: int | None = Field(default=None, ge=30, le=150)
  59. username: str | None = None
  60. password: str | None = None
  61. # Power alerts
  62. power_alert_enabled: bool | None = None
  63. power_alert_high: float | None = Field(default=None, ge=0, le=5000)
  64. power_alert_low: float | None = Field(default=None, ge=0, le=5000)
  65. # Schedule
  66. schedule_enabled: bool | None = None
  67. schedule_on_time: str | None = Field(default=None, pattern=r"^([01]\d|2[0-3]):[0-5]\d$")
  68. schedule_off_time: str | None = Field(default=None, pattern=r"^([01]\d|2[0-3]):[0-5]\d$")
  69. # Switchbar visibility
  70. show_in_switchbar: bool | None = None
  71. class SmartPlugResponse(SmartPlugBase):
  72. id: int
  73. last_state: str | None = None
  74. last_checked: datetime | None = None
  75. auto_off_executed: bool = False # True when auto-off was triggered after print
  76. power_alert_last_triggered: datetime | None = None
  77. created_at: datetime
  78. updated_at: datetime
  79. class Config:
  80. from_attributes = True
  81. class SmartPlugControl(BaseModel):
  82. action: Literal["on", "off", "toggle"]
  83. class SmartPlugEnergy(BaseModel):
  84. """Energy monitoring data from a smart plug."""
  85. power: float | None = None # Current watts
  86. voltage: float | None = None # Volts
  87. current: float | None = None # Amps
  88. today: float | None = None # kWh used today
  89. yesterday: float | None = None # kWh used yesterday
  90. total: float | None = None # Total kWh
  91. factor: float | None = None # Power factor (0-1)
  92. apparent_power: float | None = None # VA
  93. reactive_power: float | None = None # VAr
  94. class SmartPlugStatus(BaseModel):
  95. state: str | None = None # "ON", "OFF", or None if unreachable
  96. reachable: bool = True
  97. device_name: str | None = None
  98. energy: SmartPlugEnergy | None = None # Energy data if available
  99. class SmartPlugTestConnection(BaseModel):
  100. ip_address: str = Field(..., pattern=r"^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$")
  101. username: str | None = None
  102. password: str | None = None
  103. # Home Assistant schemas
  104. class HATestConnectionRequest(BaseModel):
  105. """Request to test Home Assistant connection."""
  106. url: str = Field(..., min_length=1)
  107. token: str = Field(..., min_length=1)
  108. class HATestConnectionResponse(BaseModel):
  109. """Response from HA connection test."""
  110. success: bool
  111. message: str | None = None
  112. error: str | None = None
  113. class HAEntity(BaseModel):
  114. """A Home Assistant entity that can be used as a smart plug."""
  115. entity_id: str
  116. friendly_name: str
  117. state: str | None = None
  118. domain: str # "switch", "light", "input_boolean"
  119. class HASensorEntity(BaseModel):
  120. """A Home Assistant sensor entity for energy monitoring."""
  121. entity_id: str
  122. friendly_name: str
  123. state: str | None = None
  124. unit_of_measurement: str | None = None # "W", "kW", "kWh", "Wh"