smart_plug.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  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", "mqtt", "rest"] = "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|script)\.[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. # MQTT fields (required when plug_type="mqtt")
  18. # Legacy field - kept for backward compatibility
  19. mqtt_topic: str | None = Field(default=None, max_length=200) # Deprecated, use mqtt_power_topic
  20. # Power monitoring
  21. mqtt_power_topic: str | None = Field(default=None, max_length=200) # Topic for power data
  22. mqtt_power_path: str | None = Field(default=None, max_length=100) # e.g., "power_l1" or "data.power"
  23. mqtt_power_multiplier: float = Field(default=1.0, ge=0.0001, le=10000) # Unit conversion for power
  24. # Energy monitoring
  25. mqtt_energy_topic: str | None = Field(default=None, max_length=200) # Topic for energy data
  26. mqtt_energy_path: str | None = Field(default=None, max_length=100) # e.g., "energy_l1"
  27. mqtt_energy_multiplier: float = Field(default=1.0, ge=0.0001, le=10000) # Unit conversion for energy
  28. # State monitoring
  29. mqtt_state_topic: str | None = Field(default=None, max_length=200) # Topic for state data
  30. mqtt_state_path: str | None = Field(default=None, max_length=100) # e.g., "state_l1" for ON/OFF
  31. mqtt_state_on_value: str | None = Field(
  32. default=None, max_length=50
  33. ) # What value means "ON" (e.g., "ON", "true", "1")
  34. # Legacy multiplier - kept for backward compatibility
  35. mqtt_multiplier: float = Field(default=1.0, ge=0.0001, le=10000) # Deprecated, use mqtt_power_multiplier
  36. # REST/Webhook fields (required when plug_type="rest")
  37. rest_on_url: str | None = Field(default=None, max_length=500)
  38. rest_on_body: str | None = None
  39. rest_off_url: str | None = Field(default=None, max_length=500)
  40. rest_off_body: str | None = None
  41. rest_method: Literal["GET", "POST", "PUT", "PATCH"] | None = None
  42. rest_headers: str | None = None # JSON string of custom headers
  43. rest_status_url: str | None = Field(default=None, max_length=500)
  44. rest_status_path: str | None = Field(default=None, max_length=200)
  45. rest_status_on_value: str | None = Field(default=None, max_length=50)
  46. rest_power_url: str | None = Field(default=None, max_length=500)
  47. rest_power_path: str | None = Field(default=None, max_length=200)
  48. rest_power_multiplier: float = Field(default=1.0, ge=0.0001, le=10000)
  49. rest_energy_url: str | None = Field(default=None, max_length=500)
  50. rest_energy_path: str | None = Field(default=None, max_length=200)
  51. rest_energy_multiplier: float = Field(default=1.0, ge=0.0001, le=10000)
  52. printer_id: int | None = None
  53. enabled: bool = True
  54. auto_on: bool = True
  55. auto_off: bool = True
  56. auto_off_persistent: bool = False
  57. off_delay_mode: Literal["time", "temperature"] = "time"
  58. off_delay_minutes: int = Field(default=5, ge=0, le=60)
  59. off_temp_threshold: int = Field(default=70, ge=30, le=150)
  60. # #1349: auto-off after AMS drying completes. Independent of `auto_off`
  61. # (print-finish). Fires whenever any AMS on the linked printer finishes
  62. # a dry cycle.
  63. auto_off_after_drying: bool = False
  64. off_delay_after_drying_minutes: int = Field(default=10, ge=0, le=120)
  65. # Power alerts
  66. power_alert_enabled: bool = False
  67. power_alert_high: float | None = Field(default=None, ge=0, le=5000) # Alert when power > this (watts)
  68. power_alert_low: float | None = Field(default=None, ge=0, le=5000) # Alert when power < this (watts)
  69. # Schedule
  70. schedule_enabled: bool = False
  71. schedule_on_time: str | None = Field(default=None, pattern=r"^([01]\d|2[0-3]):[0-5]\d$") # HH:MM format
  72. schedule_off_time: str | None = Field(default=None, pattern=r"^([01]\d|2[0-3]):[0-5]\d$") # HH:MM format
  73. # Visibility options
  74. show_in_switchbar: bool = False
  75. show_on_printer_card: bool = True # For scripts: show on printer card
  76. @model_validator(mode="after")
  77. def validate_plug_type_fields(self) -> "SmartPlugBase":
  78. if self.plug_type == "tasmota" and not self.ip_address:
  79. raise ValueError("ip_address is required for Tasmota plugs")
  80. if self.plug_type == "homeassistant" and not self.ha_entity_id:
  81. raise ValueError("ha_entity_id is required for Home Assistant plugs")
  82. if self.plug_type == "mqtt":
  83. # Determine the effective power topic (new field takes priority, fall back to legacy)
  84. power_topic = self.mqtt_power_topic or self.mqtt_topic
  85. # Path is optional - if not set, raw MQTT payload value will be used
  86. has_power = bool(power_topic)
  87. has_energy = bool(self.mqtt_energy_topic)
  88. has_state = bool(self.mqtt_state_topic)
  89. # At least one data source must be configured (path is optional)
  90. if not has_power and not has_energy and not has_state:
  91. raise ValueError("At least one MQTT topic must be configured for power, energy, or state monitoring")
  92. if self.plug_type == "rest":
  93. if not self.rest_on_url and not self.rest_off_url:
  94. raise ValueError("At least one of ON URL or OFF URL is required for REST plugs")
  95. return self
  96. class SmartPlugCreate(SmartPlugBase):
  97. pass
  98. class SmartPlugUpdate(BaseModel):
  99. name: str | None = None
  100. plug_type: Literal["tasmota", "homeassistant", "mqtt", "rest"] | None = None
  101. ip_address: str | None = None
  102. ha_entity_id: str | None = None
  103. # Home Assistant energy sensor entities (optional)
  104. ha_power_entity: str | None = None
  105. ha_energy_today_entity: str | None = None
  106. ha_energy_total_entity: str | None = None
  107. # MQTT fields (legacy)
  108. mqtt_topic: str | None = None
  109. mqtt_multiplier: float | None = Field(default=None, ge=0.0001, le=10000)
  110. # MQTT power fields
  111. mqtt_power_topic: str | None = None
  112. mqtt_power_path: str | None = None
  113. mqtt_power_multiplier: float | None = Field(default=None, ge=0.0001, le=10000)
  114. # MQTT energy fields
  115. mqtt_energy_topic: str | None = None
  116. mqtt_energy_path: str | None = None
  117. mqtt_energy_multiplier: float | None = Field(default=None, ge=0.0001, le=10000)
  118. # MQTT state fields
  119. mqtt_state_topic: str | None = None
  120. mqtt_state_path: str | None = None
  121. mqtt_state_on_value: str | None = None
  122. # REST fields
  123. rest_on_url: str | None = None
  124. rest_on_body: str | None = None
  125. rest_off_url: str | None = None
  126. rest_off_body: str | None = None
  127. rest_method: Literal["GET", "POST", "PUT", "PATCH"] | None = None
  128. rest_headers: str | None = None
  129. rest_status_url: str | None = None
  130. rest_status_path: str | None = None
  131. rest_status_on_value: str | None = None
  132. rest_power_url: str | None = None
  133. rest_power_path: str | None = None
  134. rest_power_multiplier: float | None = Field(default=None, ge=0.0001, le=10000)
  135. rest_energy_url: str | None = None
  136. rest_energy_path: str | None = None
  137. rest_energy_multiplier: float | None = Field(default=None, ge=0.0001, le=10000)
  138. printer_id: int | None = None
  139. enabled: bool | None = None
  140. auto_on: bool | None = None
  141. auto_off: bool | None = None
  142. auto_off_persistent: bool | None = None
  143. off_delay_mode: Literal["time", "temperature"] | None = None
  144. off_delay_minutes: int | None = Field(default=None, ge=0, le=60)
  145. off_temp_threshold: int | None = Field(default=None, ge=30, le=150)
  146. # #1349: per-plug drying auto-off.
  147. auto_off_after_drying: bool | None = None
  148. off_delay_after_drying_minutes: int | None = Field(default=None, ge=0, le=120)
  149. username: str | None = None
  150. password: str | None = None
  151. # Power alerts
  152. power_alert_enabled: bool | None = None
  153. power_alert_high: float | None = Field(default=None, ge=0, le=5000)
  154. power_alert_low: float | None = Field(default=None, ge=0, le=5000)
  155. # Schedule
  156. schedule_enabled: bool | None = None
  157. schedule_on_time: str | None = Field(default=None, pattern=r"^([01]\d|2[0-3]):[0-5]\d$")
  158. schedule_off_time: str | None = Field(default=None, pattern=r"^([01]\d|2[0-3]):[0-5]\d$")
  159. # Visibility options
  160. show_in_switchbar: bool | None = None
  161. show_on_printer_card: bool | None = None
  162. class SmartPlugResponse(SmartPlugBase):
  163. id: int
  164. last_state: str | None = None
  165. last_checked: datetime | None = None
  166. auto_off_executed: bool = False # True when auto-off was triggered after print
  167. power_alert_last_triggered: datetime | None = None
  168. created_at: datetime
  169. updated_at: datetime
  170. class Config:
  171. from_attributes = True
  172. class SmartPlugControl(BaseModel):
  173. action: Literal["on", "off", "toggle"]
  174. class SmartPlugEnergy(BaseModel):
  175. """Energy monitoring data from a smart plug."""
  176. power: float | None = None # Current watts
  177. voltage: float | None = None # Volts
  178. current: float | None = None # Amps
  179. today: float | None = None # kWh used today
  180. yesterday: float | None = None # kWh used yesterday
  181. total: float | None = None # Total kWh
  182. factor: float | None = None # Power factor (0-1)
  183. apparent_power: float | None = None # VA
  184. reactive_power: float | None = None # VAr
  185. class SmartPlugStatus(BaseModel):
  186. state: str | None = None # "ON", "OFF", or None if unreachable
  187. reachable: bool = True
  188. device_name: str | None = None
  189. energy: SmartPlugEnergy | None = None # Energy data if available
  190. class SmartPlugTestConnection(BaseModel):
  191. ip_address: str = Field(..., pattern=r"^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}$")
  192. username: str | None = None
  193. password: str | None = None
  194. # Home Assistant schemas
  195. class HATestConnectionRequest(BaseModel):
  196. """Request to test Home Assistant connection."""
  197. url: str = Field(..., min_length=1)
  198. token: str = Field(..., min_length=1)
  199. class HATestConnectionResponse(BaseModel):
  200. """Response from HA connection test."""
  201. success: bool
  202. message: str | None = None
  203. error: str | None = None
  204. class HAEntity(BaseModel):
  205. """A Home Assistant entity that can be used as a smart plug."""
  206. entity_id: str
  207. friendly_name: str
  208. state: str | None = None
  209. domain: str # "switch", "light", "input_boolean", "script"
  210. class HASensorEntity(BaseModel):
  211. """A Home Assistant sensor entity for energy monitoring."""
  212. entity_id: str
  213. friendly_name: str
  214. state: str | None = None
  215. unit_of_measurement: str | None = None # "W", "kW", "kWh", "Wh"
  216. class RESTTestConnectionRequest(BaseModel):
  217. """Request to test a REST smart plug connection."""
  218. url: str = Field(..., min_length=1)
  219. method: str = Field(default="GET")
  220. headers: str | None = None # JSON string of custom headers
  221. class RESTTestConnectionResponse(BaseModel):
  222. """Response from REST connection test."""
  223. success: bool
  224. error: str | None = None