printer.py 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  1. from datetime import datetime
  2. from sqlalchemy import Boolean, DateTime, Float, String, func
  3. from sqlalchemy.orm import Mapped, mapped_column, relationship
  4. from backend.app.core.database import Base
  5. class Printer(Base):
  6. __tablename__ = "printers"
  7. id: Mapped[int] = mapped_column(primary_key=True)
  8. name: Mapped[str] = mapped_column(String(100))
  9. serial_number: Mapped[str] = mapped_column(String(50), unique=True)
  10. ip_address: Mapped[str] = mapped_column(String(253))
  11. access_code: Mapped[str] = mapped_column(String(20))
  12. model: Mapped[str | None] = mapped_column(String(50))
  13. location: Mapped[str | None] = mapped_column(String(100)) # Group/location name
  14. nozzle_count: Mapped[int] = mapped_column(default=1) # 1 or 2, auto-detected from MQTT
  15. is_active: Mapped[bool] = mapped_column(Boolean, default=True)
  16. auto_archive: Mapped[bool] = mapped_column(Boolean, default=True)
  17. print_hours_offset: Mapped[float] = mapped_column(Float, default=0.0) # Baseline hours to add
  18. runtime_seconds: Mapped[int] = mapped_column(default=0) # Accumulated active runtime (RUNNING/PAUSE states)
  19. last_runtime_update: Mapped[datetime | None] = mapped_column(
  20. DateTime, nullable=True
  21. ) # Last time runtime was updated
  22. # External camera configuration
  23. external_camera_url: Mapped[str | None] = mapped_column(String(500), nullable=True)
  24. external_camera_type: Mapped[str | None] = mapped_column(String(20), nullable=True) # mjpeg, rtsp, snapshot
  25. external_camera_enabled: Mapped[bool] = mapped_column(Boolean, default=False)
  26. # Optional single-frame snapshot URL — when set, used for snapshot / finish-photo
  27. # / timelapse / plate-detect captures instead of opening the live stream and
  28. # skipping a warm-up frame. Bypasses MJPEG warm-up issues on sources that
  29. # expose a dedicated frame endpoint (e.g. go2rtc's /api/frame.jpeg). #1177.
  30. external_camera_snapshot_url: Mapped[str | None] = mapped_column(String(500), nullable=True)
  31. camera_rotation: Mapped[int] = mapped_column(default=0) # 0, 90, 180, 270 degrees
  32. # Plate detection - check if build plate is empty before starting print
  33. plate_detection_enabled: Mapped[bool] = mapped_column(Boolean, default=False)
  34. # ROI for plate detection (percentages: 0.0-1.0)
  35. plate_detection_roi_x: Mapped[float | None] = mapped_column(Float, nullable=True) # X start %
  36. plate_detection_roi_y: Mapped[float | None] = mapped_column(Float, nullable=True) # Y start %
  37. plate_detection_roi_w: Mapped[float | None] = mapped_column(Float, nullable=True) # Width %
  38. plate_detection_roi_h: Mapped[float | None] = mapped_column(Float, nullable=True) # Height %
  39. # Queue: True after a print finishes/fails, until user acknowledges the plate is cleared.
  40. # Persisted so the gate survives crashes and power cycles (issue #961).
  41. awaiting_plate_clear: Mapped[bool] = mapped_column(Boolean, default=False)
  42. created_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now())
  43. updated_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now(), onupdate=func.now())
  44. # Relationships
  45. archives: Mapped[list["PrintArchive"]] = relationship(back_populates="printer", cascade="all, delete-orphan")
  46. smart_plugs: Mapped[list["SmartPlug"]] = relationship(back_populates="printer")
  47. notification_providers: Mapped[list["NotificationProvider"]] = relationship(back_populates="printer")
  48. maintenance_items: Mapped[list["PrinterMaintenance"]] = relationship(
  49. back_populates="printer", cascade="all, delete-orphan"
  50. )
  51. kprofile_notes: Mapped[list["KProfileNote"]] = relationship(back_populates="printer", cascade="all, delete-orphan")
  52. ams_history: Mapped[list["AMSSensorHistory"]] = relationship(back_populates="printer", cascade="all, delete-orphan")
  53. from backend.app.models.ams_history import AMSSensorHistory # noqa: E402
  54. from backend.app.models.archive import PrintArchive # noqa: E402
  55. from backend.app.models.kprofile_note import KProfileNote # noqa: E402
  56. from backend.app.models.maintenance import PrinterMaintenance # noqa: E402
  57. from backend.app.models.notification import NotificationProvider # noqa: E402
  58. from backend.app.models.smart_plug import SmartPlug # noqa: E402