smart_plugs.py 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680
  1. """API routes for smart plug management."""
  2. import logging
  3. from datetime import datetime, timedelta
  4. from fastapi import APIRouter, Body, Depends, HTTPException
  5. from pydantic import BaseModel
  6. from sqlalchemy import select
  7. from sqlalchemy.ext.asyncio import AsyncSession
  8. from backend.app.api.routes.settings import get_setting
  9. from backend.app.core.database import get_db
  10. from backend.app.models.printer import Printer
  11. from backend.app.models.smart_plug import SmartPlug
  12. from backend.app.schemas.smart_plug import (
  13. HAEntity,
  14. HASensorEntity,
  15. HATestConnectionRequest,
  16. HATestConnectionResponse,
  17. SmartPlugControl,
  18. SmartPlugCreate,
  19. SmartPlugEnergy,
  20. SmartPlugResponse,
  21. SmartPlugStatus,
  22. SmartPlugTestConnection,
  23. SmartPlugUpdate,
  24. )
  25. from backend.app.services.discovery import tasmota_scanner
  26. from backend.app.services.homeassistant import homeassistant_service
  27. from backend.app.services.mqtt_relay import mqtt_relay
  28. from backend.app.services.notification_service import notification_service
  29. from backend.app.services.printer_manager import printer_manager
  30. from backend.app.services.tasmota import tasmota_service
  31. logger = logging.getLogger(__name__)
  32. router = APIRouter(prefix="/smart-plugs", tags=["smart-plugs"])
  33. @router.get("/", response_model=list[SmartPlugResponse])
  34. async def list_smart_plugs(db: AsyncSession = Depends(get_db)):
  35. """List all smart plugs."""
  36. result = await db.execute(select(SmartPlug).order_by(SmartPlug.name))
  37. return list(result.scalars().all())
  38. @router.post("/", response_model=SmartPlugResponse)
  39. async def create_smart_plug(
  40. data: SmartPlugCreate,
  41. db: AsyncSession = Depends(get_db),
  42. ):
  43. """Create a new smart plug."""
  44. # Validate printer_id if provided
  45. if data.printer_id:
  46. result = await db.execute(select(Printer).where(Printer.id == data.printer_id))
  47. if not result.scalar_one_or_none():
  48. raise HTTPException(400, "Printer not found")
  49. # Check if printer already has a plug assigned
  50. result = await db.execute(select(SmartPlug).where(SmartPlug.printer_id == data.printer_id))
  51. if result.scalar_one_or_none():
  52. raise HTTPException(400, "This printer already has a smart plug assigned")
  53. # For MQTT plugs, ensure MQTT broker is configured and service is connected
  54. if data.plug_type == "mqtt":
  55. # Try to configure the smart plug service if not already configured
  56. if not mqtt_relay.smart_plug_service.is_configured():
  57. # Get MQTT broker settings from database
  58. mqtt_broker = await get_setting(db, "mqtt_broker") or ""
  59. if not mqtt_broker:
  60. raise HTTPException(
  61. 400,
  62. "MQTT broker not configured. Please set MQTT broker address in Settings → Network → MQTT Publishing.",
  63. )
  64. # Configure the smart plug service with broker settings
  65. mqtt_settings = {
  66. "mqtt_enabled": True, # Enable for smart plug subscription
  67. "mqtt_broker": mqtt_broker,
  68. "mqtt_port": int(await get_setting(db, "mqtt_port") or "1883"),
  69. "mqtt_username": await get_setting(db, "mqtt_username") or "",
  70. "mqtt_password": await get_setting(db, "mqtt_password") or "",
  71. "mqtt_use_tls": (await get_setting(db, "mqtt_use_tls") or "false") == "true",
  72. }
  73. await mqtt_relay.smart_plug_service.configure(mqtt_settings)
  74. # Check if connection succeeded
  75. if not mqtt_relay.smart_plug_service.is_configured():
  76. raise HTTPException(
  77. 400,
  78. f"Failed to connect to MQTT broker at {mqtt_broker}. Please check your MQTT settings.",
  79. )
  80. plug = SmartPlug(**data.model_dump())
  81. db.add(plug)
  82. await db.commit()
  83. await db.refresh(plug)
  84. # Subscribe MQTT plugs to their topics
  85. if plug.plug_type == "mqtt":
  86. # Determine effective topics (new fields take priority, fall back to legacy)
  87. power_topic = plug.mqtt_power_topic or plug.mqtt_topic
  88. energy_topic = plug.mqtt_energy_topic
  89. state_topic = plug.mqtt_state_topic
  90. # Only subscribe if at least one topic is configured
  91. if power_topic or energy_topic or state_topic:
  92. mqtt_relay.smart_plug_service.subscribe(
  93. plug_id=plug.id,
  94. # Power source (path is optional)
  95. power_topic=power_topic,
  96. power_path=plug.mqtt_power_path,
  97. power_multiplier=plug.mqtt_power_multiplier or plug.mqtt_multiplier or 1.0,
  98. # Energy source (path is optional)
  99. energy_topic=energy_topic,
  100. energy_path=plug.mqtt_energy_path,
  101. energy_multiplier=plug.mqtt_energy_multiplier or plug.mqtt_multiplier or 1.0,
  102. # State source (path is optional)
  103. state_topic=state_topic,
  104. state_path=plug.mqtt_state_path,
  105. state_on_value=plug.mqtt_state_on_value,
  106. )
  107. topics = [t for t in [power_topic, energy_topic, state_topic] if t]
  108. logger.info(f"Created MQTT plug '{plug.name}' subscribed to {', '.join(set(topics))}")
  109. elif plug.plug_type == "homeassistant":
  110. logger.info(f"Created Home Assistant plug '{plug.name}' ({plug.ha_entity_id})")
  111. else:
  112. logger.info(f"Created Tasmota plug '{plug.name}' at {plug.ip_address}")
  113. return plug
  114. @router.get("/by-printer/{printer_id}", response_model=SmartPlugResponse | None)
  115. async def get_smart_plug_by_printer(printer_id: int, db: AsyncSession = Depends(get_db)):
  116. """Get the smart plug assigned to a printer."""
  117. result = await db.execute(select(SmartPlug).where(SmartPlug.printer_id == printer_id))
  118. plug = result.scalar_one_or_none()
  119. if not plug:
  120. return None
  121. return plug
  122. # Tasmota Discovery Endpoints
  123. # NOTE: These must be defined BEFORE /{plug_id} routes to avoid path conflicts
  124. class TasmotaScanRequest(BaseModel):
  125. """Request to scan for Tasmota devices."""
  126. from_ip: str | None = None # Starting IP (auto-detected if not provided)
  127. to_ip: str | None = None # Ending IP (auto-detected if not provided)
  128. timeout: float = 1.0 # Connection timeout per host
  129. def get_local_network_range() -> tuple[str, str]:
  130. """Auto-detect local network and return IP range to scan."""
  131. import socket
  132. try:
  133. # Get local IP by connecting to a public DNS (doesn't actually send data)
  134. s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
  135. s.connect(("8.8.8.8", 80))
  136. local_ip = s.getsockname()[0]
  137. s.close()
  138. # Parse IP and create range (assume /24 subnet)
  139. parts = local_ip.split(".")
  140. base = ".".join(parts[:3])
  141. from_ip = f"{base}.1"
  142. to_ip = f"{base}.254"
  143. logger.info(f"Auto-detected network: {from_ip} - {to_ip} (local IP: {local_ip})")
  144. return from_ip, to_ip
  145. except Exception as e:
  146. logger.error(f"Failed to detect local network: {e}")
  147. # Fallback to common home network
  148. return "192.168.1.1", "192.168.1.254"
  149. class TasmotaScanStatus(BaseModel):
  150. """Tasmota scan status response."""
  151. running: bool
  152. scanned: int
  153. total: int
  154. class DiscoveredTasmotaDevice(BaseModel):
  155. """Discovered Tasmota device."""
  156. ip_address: str
  157. name: str
  158. module: int | None = None
  159. state: str | None = None
  160. discovered_at: str | None = None
  161. @router.post("/discover/scan", response_model=TasmotaScanStatus)
  162. async def start_tasmota_scan(request: TasmotaScanRequest | None = Body(default=None)):
  163. """Start an IP range scan for Tasmota devices.
  164. Auto-detects local network if no IP range provided.
  165. """
  166. import asyncio
  167. # Auto-detect network
  168. from_ip, to_ip = get_local_network_range()
  169. timeout = request.timeout if request else 1.0
  170. # Start scan in background
  171. asyncio.create_task(tasmota_scanner.scan_range(from_ip, to_ip, timeout))
  172. # Return immediate status
  173. scanned, total = tasmota_scanner.progress
  174. return TasmotaScanStatus(
  175. running=tasmota_scanner.is_running,
  176. scanned=scanned,
  177. total=total,
  178. )
  179. @router.get("/discover/status", response_model=TasmotaScanStatus)
  180. async def get_tasmota_scan_status():
  181. """Get the current Tasmota scan status."""
  182. scanned, total = tasmota_scanner.progress
  183. return TasmotaScanStatus(
  184. running=tasmota_scanner.is_running,
  185. scanned=scanned,
  186. total=total,
  187. )
  188. @router.post("/discover/stop", response_model=TasmotaScanStatus)
  189. async def stop_tasmota_scan():
  190. """Stop the current Tasmota scan."""
  191. tasmota_scanner.stop()
  192. scanned, total = tasmota_scanner.progress
  193. return TasmotaScanStatus(
  194. running=tasmota_scanner.is_running,
  195. scanned=scanned,
  196. total=total,
  197. )
  198. @router.get("/discover/devices", response_model=list[DiscoveredTasmotaDevice])
  199. async def get_discovered_tasmota_devices():
  200. """Get list of discovered Tasmota devices."""
  201. return [
  202. DiscoveredTasmotaDevice(
  203. ip_address=d["ip_address"],
  204. name=d["name"],
  205. module=d.get("module"),
  206. state=d.get("state"),
  207. discovered_at=d.get("discovered_at"),
  208. )
  209. for d in tasmota_scanner.discovered_devices
  210. ]
  211. # Home Assistant Discovery Endpoints
  212. @router.post("/ha/test-connection", response_model=HATestConnectionResponse)
  213. async def test_ha_connection(request: HATestConnectionRequest):
  214. """Test connection to Home Assistant."""
  215. result = await homeassistant_service.test_connection(request.url, request.token)
  216. return HATestConnectionResponse(**result)
  217. @router.get("/ha/entities", response_model=list[HAEntity])
  218. async def list_ha_entities(
  219. db: AsyncSession = Depends(get_db),
  220. search: str | None = None,
  221. ):
  222. """List available Home Assistant entities.
  223. By default, returns switch/light/input_boolean entities.
  224. When search is provided, searches ALL entities by entity_id or friendly_name.
  225. Requires HA connection settings to be configured in Settings.
  226. """
  227. ha_url = await get_setting(db, "ha_url") or ""
  228. ha_token = await get_setting(db, "ha_token") or ""
  229. if not ha_url or not ha_token:
  230. raise HTTPException(
  231. 400, "Home Assistant not configured. Please set HA URL and token in Settings → Network → Home Assistant."
  232. )
  233. entities = await homeassistant_service.list_entities(ha_url, ha_token, search)
  234. return [HAEntity(**e) for e in entities]
  235. @router.get("/ha/sensors", response_model=list[HASensorEntity])
  236. async def list_ha_sensor_entities(db: AsyncSession = Depends(get_db)):
  237. """List available Home Assistant sensor entities for energy monitoring.
  238. Returns sensors with power/energy units (W, kW, kWh, Wh).
  239. Requires HA connection settings to be configured in Settings.
  240. """
  241. ha_url = await get_setting(db, "ha_url") or ""
  242. ha_token = await get_setting(db, "ha_token") or ""
  243. if not ha_url or not ha_token:
  244. raise HTTPException(
  245. 400, "Home Assistant not configured. Please set HA URL and token in Settings → Network → Home Assistant."
  246. )
  247. sensors = await homeassistant_service.list_sensor_entities(ha_url, ha_token)
  248. return [HASensorEntity(**s) for s in sensors]
  249. @router.get("/{plug_id}", response_model=SmartPlugResponse)
  250. async def get_smart_plug(plug_id: int, db: AsyncSession = Depends(get_db)):
  251. """Get a specific smart plug."""
  252. result = await db.execute(select(SmartPlug).where(SmartPlug.id == plug_id))
  253. plug = result.scalar_one_or_none()
  254. if not plug:
  255. raise HTTPException(404, "Smart plug not found")
  256. return plug
  257. @router.patch("/{plug_id}", response_model=SmartPlugResponse)
  258. async def update_smart_plug(
  259. plug_id: int,
  260. data: SmartPlugUpdate,
  261. db: AsyncSession = Depends(get_db),
  262. ):
  263. """Update a smart plug."""
  264. result = await db.execute(select(SmartPlug).where(SmartPlug.id == plug_id))
  265. plug = result.scalar_one_or_none()
  266. if not plug:
  267. raise HTTPException(404, "Smart plug not found")
  268. update_data = data.model_dump(exclude_unset=True)
  269. # Validate new printer_id if being changed
  270. if "printer_id" in update_data and update_data["printer_id"]:
  271. new_printer_id = update_data["printer_id"]
  272. # Check printer exists
  273. result = await db.execute(select(Printer).where(Printer.id == new_printer_id))
  274. if not result.scalar_one_or_none():
  275. raise HTTPException(400, "Printer not found")
  276. # Check if that printer already has a different plug assigned
  277. result = await db.execute(
  278. select(SmartPlug).where(
  279. SmartPlug.printer_id == new_printer_id,
  280. SmartPlug.id != plug_id,
  281. )
  282. )
  283. if result.scalar_one_or_none():
  284. raise HTTPException(400, "This printer already has a smart plug assigned")
  285. # Track old MQTT settings for comparison
  286. old_plug_type = plug.plug_type
  287. old_mqtt_config = {
  288. "power_topic": plug.mqtt_power_topic or plug.mqtt_topic,
  289. "power_path": plug.mqtt_power_path,
  290. "power_multiplier": plug.mqtt_power_multiplier,
  291. "energy_topic": plug.mqtt_energy_topic or plug.mqtt_topic,
  292. "energy_path": plug.mqtt_energy_path,
  293. "energy_multiplier": plug.mqtt_energy_multiplier,
  294. "state_topic": plug.mqtt_state_topic or plug.mqtt_topic,
  295. "state_path": plug.mqtt_state_path,
  296. "state_on_value": plug.mqtt_state_on_value,
  297. }
  298. for field, value in update_data.items():
  299. setattr(plug, field, value)
  300. await db.commit()
  301. await db.refresh(plug)
  302. # Handle MQTT subscription changes
  303. if old_plug_type == "mqtt" and plug.plug_type != "mqtt":
  304. # Changed away from MQTT - unsubscribe
  305. mqtt_relay.smart_plug_service.unsubscribe(plug.id)
  306. elif plug.plug_type == "mqtt":
  307. # Check if any MQTT config changed
  308. new_mqtt_config = {
  309. "power_topic": plug.mqtt_power_topic or plug.mqtt_topic,
  310. "power_path": plug.mqtt_power_path,
  311. "power_multiplier": plug.mqtt_power_multiplier,
  312. "energy_topic": plug.mqtt_energy_topic or plug.mqtt_topic,
  313. "energy_path": plug.mqtt_energy_path,
  314. "energy_multiplier": plug.mqtt_energy_multiplier,
  315. "state_topic": plug.mqtt_state_topic or plug.mqtt_topic,
  316. "state_path": plug.mqtt_state_path,
  317. "state_on_value": plug.mqtt_state_on_value,
  318. }
  319. mqtt_changed = old_plug_type != "mqtt" or old_mqtt_config != new_mqtt_config
  320. if mqtt_changed:
  321. # Unsubscribe from old topics first
  322. if old_plug_type == "mqtt":
  323. mqtt_relay.smart_plug_service.unsubscribe(plug.id)
  324. # Subscribe to new topics
  325. power_topic = plug.mqtt_power_topic or plug.mqtt_topic
  326. energy_topic = plug.mqtt_energy_topic
  327. state_topic = plug.mqtt_state_topic
  328. # Only subscribe if at least one topic is configured
  329. if power_topic or energy_topic or state_topic:
  330. mqtt_relay.smart_plug_service.subscribe(
  331. plug_id=plug.id,
  332. # Power source (path is optional)
  333. power_topic=power_topic,
  334. power_path=plug.mqtt_power_path,
  335. power_multiplier=plug.mqtt_power_multiplier or plug.mqtt_multiplier or 1.0,
  336. # Energy source (path is optional)
  337. energy_topic=energy_topic,
  338. energy_path=plug.mqtt_energy_path,
  339. energy_multiplier=plug.mqtt_energy_multiplier or plug.mqtt_multiplier or 1.0,
  340. # State source (path is optional)
  341. state_topic=state_topic,
  342. state_path=plug.mqtt_state_path,
  343. state_on_value=plug.mqtt_state_on_value,
  344. )
  345. logger.info(f"Updated smart plug '{plug.name}'")
  346. return plug
  347. @router.delete("/{plug_id}")
  348. async def delete_smart_plug(plug_id: int, db: AsyncSession = Depends(get_db)):
  349. """Delete a smart plug."""
  350. result = await db.execute(select(SmartPlug).where(SmartPlug.id == plug_id))
  351. plug = result.scalar_one_or_none()
  352. if not plug:
  353. raise HTTPException(404, "Smart plug not found")
  354. plug_name = plug.name
  355. plug_type = plug.plug_type
  356. # Unsubscribe MQTT plug before deletion
  357. if plug_type == "mqtt":
  358. mqtt_relay.smart_plug_service.unsubscribe(plug_id)
  359. await db.delete(plug)
  360. await db.commit()
  361. logger.info(f"Deleted smart plug '{plug_name}'")
  362. return {"message": "Smart plug deleted"}
  363. async def _get_service_for_plug(plug: SmartPlug, db: AsyncSession):
  364. """Get the appropriate service for the plug type.
  365. For HA plugs, configures the service with current settings from DB.
  366. """
  367. if plug.plug_type == "homeassistant":
  368. # Configure HA service with current settings
  369. ha_url = await get_setting(db, "ha_url") or ""
  370. ha_token = await get_setting(db, "ha_token") or ""
  371. homeassistant_service.configure(ha_url, ha_token)
  372. return homeassistant_service
  373. return tasmota_service
  374. @router.post("/{plug_id}/control")
  375. async def control_smart_plug(
  376. plug_id: int,
  377. control: SmartPlugControl,
  378. db: AsyncSession = Depends(get_db),
  379. ):
  380. """Manual control: on/off/toggle."""
  381. result = await db.execute(select(SmartPlug).where(SmartPlug.id == plug_id))
  382. plug = result.scalar_one_or_none()
  383. if not plug:
  384. raise HTTPException(404, "Smart plug not found")
  385. # MQTT plugs are monitor-only - cannot control them
  386. if plug.plug_type == "mqtt":
  387. raise HTTPException(
  388. 400,
  389. "MQTT plugs are monitor-only. Use your MQTT broker or home automation system to control them.",
  390. )
  391. service = await _get_service_for_plug(plug, db)
  392. if control.action == "on":
  393. success = await service.turn_on(plug)
  394. expected_state = "ON"
  395. elif control.action == "off":
  396. success = await service.turn_off(plug)
  397. expected_state = "OFF"
  398. elif control.action == "toggle":
  399. success = await service.toggle(plug)
  400. expected_state = None # Unknown after toggle
  401. else:
  402. raise HTTPException(400, f"Invalid action: {control.action}")
  403. if not success:
  404. raise HTTPException(503, "Failed to communicate with device")
  405. # Update last state and reset auto_off_executed when turning on
  406. if expected_state:
  407. plug.last_state = expected_state
  408. if expected_state == "ON":
  409. plug.auto_off_executed = False # Reset flag when manually turning on
  410. elif expected_state == "OFF" and plug.printer_id:
  411. # Mark printer offline immediately for faster UI update
  412. printer_manager.mark_printer_offline(plug.printer_id)
  413. plug.last_checked = datetime.utcnow()
  414. await db.commit()
  415. # MQTT relay - publish smart plug state change
  416. if expected_state:
  417. try:
  418. from backend.app.services.mqtt_relay import mqtt_relay
  419. # Get printer name if linked
  420. printer_name = None
  421. if plug.printer_id:
  422. result = await db.execute(select(Printer).where(Printer.id == plug.printer_id))
  423. printer = result.scalar_one_or_none()
  424. printer_name = printer.name if printer else None
  425. await mqtt_relay.on_smart_plug_state(
  426. plug_id=plug.id,
  427. plug_name=plug.name,
  428. state="on" if expected_state == "ON" else "off",
  429. printer_id=plug.printer_id,
  430. printer_name=printer_name,
  431. )
  432. except Exception:
  433. pass # Don't fail if MQTT fails
  434. return {"success": True, "action": control.action}
  435. @router.get("/{plug_id}/status", response_model=SmartPlugStatus)
  436. async def get_plug_status(plug_id: int, db: AsyncSession = Depends(get_db)):
  437. """Get current plug status from device including energy data."""
  438. result = await db.execute(select(SmartPlug).where(SmartPlug.id == plug_id))
  439. plug = result.scalar_one_or_none()
  440. if not plug:
  441. raise HTTPException(404, "Smart plug not found")
  442. # Handle MQTT plugs - get data from subscription service
  443. if plug.plug_type == "mqtt":
  444. data = mqtt_relay.smart_plug_service.get_plug_data(plug_id)
  445. is_reachable = mqtt_relay.smart_plug_service.is_reachable(plug_id)
  446. if data:
  447. # Update last state in database
  448. if is_reachable and data.state:
  449. plug.last_state = data.state
  450. plug.last_checked = datetime.utcnow()
  451. await db.commit()
  452. energy_data = None
  453. if data.power is not None or data.energy is not None:
  454. energy_data = SmartPlugEnergy(
  455. power=data.power,
  456. today=data.energy,
  457. )
  458. # Check power alerts
  459. if data.power is not None:
  460. await check_power_alerts(plug, data.power, db)
  461. return SmartPlugStatus(
  462. state=data.state,
  463. reachable=is_reachable,
  464. device_name=None,
  465. energy=energy_data,
  466. )
  467. # No data received yet
  468. return SmartPlugStatus(
  469. state=None,
  470. reachable=False,
  471. device_name=None,
  472. energy=None,
  473. )
  474. # Handle Tasmota/HomeAssistant plugs
  475. service = await _get_service_for_plug(plug, db)
  476. status = await service.get_status(plug)
  477. # Update last state in database
  478. if status["reachable"]:
  479. plug.last_state = status["state"]
  480. plug.last_checked = datetime.utcnow()
  481. await db.commit()
  482. # Fetch energy data if device is reachable
  483. energy_data = None
  484. if status["reachable"]:
  485. energy = await service.get_energy(plug)
  486. if energy:
  487. energy_data = SmartPlugEnergy(**energy)
  488. # Check power alerts
  489. await check_power_alerts(plug, energy.get("power"), db)
  490. return SmartPlugStatus(
  491. state=status["state"],
  492. reachable=status["reachable"],
  493. device_name=status.get("device_name"),
  494. energy=energy_data,
  495. )
  496. async def check_power_alerts(plug: SmartPlug, current_power: float | None, db: AsyncSession):
  497. """Check if power crosses alert thresholds and send notifications."""
  498. if not plug.power_alert_enabled or current_power is None:
  499. return
  500. # Cooldown: don't alert more than once per 5 minutes
  501. cooldown_minutes = 5
  502. if plug.power_alert_last_triggered:
  503. time_since_last = datetime.utcnow() - plug.power_alert_last_triggered
  504. if time_since_last < timedelta(minutes=cooldown_minutes):
  505. return
  506. alert_triggered = False
  507. alert_type = None
  508. threshold = None
  509. # Check high threshold
  510. if plug.power_alert_high is not None and current_power > plug.power_alert_high:
  511. alert_triggered = True
  512. alert_type = "high"
  513. threshold = plug.power_alert_high
  514. # Check low threshold
  515. if plug.power_alert_low is not None and current_power < plug.power_alert_low:
  516. alert_triggered = True
  517. alert_type = "low"
  518. threshold = plug.power_alert_low
  519. if alert_triggered:
  520. plug.power_alert_last_triggered = datetime.utcnow()
  521. await db.commit()
  522. # Send notification
  523. title = f"Power Alert: {plug.name}"
  524. if alert_type == "high":
  525. message = f"Power consumption is {current_power:.1f}W, above threshold of {threshold:.1f}W"
  526. else:
  527. message = f"Power consumption is {current_power:.1f}W, below threshold of {threshold:.1f}W"
  528. logger.info(f"Power alert triggered for {plug.name}: {message}")
  529. # Use printer_error event type for power alerts (closest match)
  530. await notification_service.send_notification(
  531. event_type="printer_error",
  532. title=title,
  533. message=message,
  534. printer_id=plug.printer_id,
  535. printer_name=plug.name,
  536. context={
  537. "error_type": f"Power {alert_type.title()}",
  538. "error_detail": message,
  539. },
  540. )
  541. @router.post("/test-connection")
  542. async def test_connection(data: SmartPlugTestConnection):
  543. """Test connection to a Tasmota device."""
  544. result = await tasmota_service.test_connection(
  545. data.ip_address,
  546. data.username,
  547. data.password,
  548. )
  549. if not result["success"]:
  550. raise HTTPException(503, result.get("error", "Failed to connect to device"))
  551. return {
  552. "success": True,
  553. "state": result["state"],
  554. "device_name": result.get("device_name"),
  555. }