smart_plug.py 3.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768
  1. from datetime import datetime
  2. from sqlalchemy import Boolean, DateTime, Float, ForeignKey, Integer, String, func
  3. from sqlalchemy.orm import Mapped, mapped_column, relationship
  4. from backend.app.core.database import Base
  5. class SmartPlug(Base):
  6. """Tasmota smart plug for printer power control."""
  7. __tablename__ = "smart_plugs"
  8. id: Mapped[int] = mapped_column(primary_key=True)
  9. name: Mapped[str] = mapped_column(String(100))
  10. ip_address: Mapped[str] = mapped_column(String(45)) # IPv4/IPv6
  11. # Link to printer (1:1)
  12. printer_id: Mapped[int | None] = mapped_column(
  13. ForeignKey("printers.id", ondelete="SET NULL"), unique=True, nullable=True
  14. )
  15. # Automation settings
  16. enabled: Mapped[bool] = mapped_column(Boolean, default=True)
  17. auto_on: Mapped[bool] = mapped_column(Boolean, default=True) # Turn on at print start
  18. auto_off: Mapped[bool] = mapped_column(Boolean, default=True) # Turn off at print complete/fail
  19. # Turn-off delay mode: "time" or "temperature"
  20. off_delay_mode: Mapped[str] = mapped_column(String(20), default="time")
  21. off_delay_minutes: Mapped[int] = mapped_column(Integer, default=5) # For time mode
  22. off_temp_threshold: Mapped[int] = mapped_column(Integer, default=70) # For temp mode (°C)
  23. # Optional auth (some Tasmota configs require it)
  24. username: Mapped[str | None] = mapped_column(String(50), nullable=True)
  25. password: Mapped[str | None] = mapped_column(String(100), nullable=True)
  26. # Power alerts
  27. power_alert_enabled: Mapped[bool] = mapped_column(Boolean, default=False)
  28. power_alert_high: Mapped[float | None] = mapped_column(Float, nullable=True) # Alert when power > this (watts)
  29. power_alert_low: Mapped[float | None] = mapped_column(Float, nullable=True) # Alert when power < this (watts)
  30. power_alert_last_triggered: Mapped[datetime | None] = mapped_column(DateTime, nullable=True) # Cooldown tracking
  31. # Schedule (time-based on/off)
  32. schedule_enabled: Mapped[bool] = mapped_column(Boolean, default=False)
  33. schedule_on_time: Mapped[str | None] = mapped_column(String(5), nullable=True) # "HH:MM" format
  34. schedule_off_time: Mapped[str | None] = mapped_column(String(5), nullable=True) # "HH:MM" format
  35. # Switchbar visibility
  36. show_in_switchbar: Mapped[bool] = mapped_column(Boolean, default=False)
  37. # Status tracking
  38. last_state: Mapped[str | None] = mapped_column(String(10), nullable=True) # "ON"/"OFF"
  39. last_checked: Mapped[datetime | None] = mapped_column(DateTime, nullable=True)
  40. auto_off_executed: Mapped[bool] = mapped_column(Boolean, default=False) # True when auto-off was triggered
  41. auto_off_pending: Mapped[bool] = mapped_column(Boolean, default=False) # True when waiting for cooldown
  42. auto_off_pending_since: Mapped[datetime | None] = mapped_column(
  43. DateTime, nullable=True
  44. ) # When auto-off was scheduled
  45. # Timestamps
  46. created_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now())
  47. updated_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now(), onupdate=func.now())
  48. # Relationship
  49. printer: Mapped["Printer"] = relationship(back_populates="smart_plug")
  50. from backend.app.models.printer import Printer # noqa: E402