maintenance.py 3.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. """Maintenance tracking models."""
  2. from datetime import datetime
  3. from sqlalchemy import Boolean, DateTime, Float, ForeignKey, String, Text, func
  4. from sqlalchemy.orm import Mapped, mapped_column, relationship
  5. from backend.app.core.database import Base
  6. class MaintenanceType(Base):
  7. """Defines a type of maintenance task with default interval."""
  8. __tablename__ = "maintenance_types"
  9. id: Mapped[int] = mapped_column(primary_key=True)
  10. name: Mapped[str] = mapped_column(String(100))
  11. description: Mapped[str | None] = mapped_column(Text)
  12. default_interval_hours: Mapped[float] = mapped_column(Float, default=100.0)
  13. # Interval type: "hours" (print hours) or "days" (calendar days)
  14. interval_type: Mapped[str] = mapped_column(String(20), default="hours")
  15. icon: Mapped[str | None] = mapped_column(String(50)) # Icon name for UI
  16. wiki_url: Mapped[str | None] = mapped_column(String(500)) # Documentation link
  17. is_system: Mapped[bool] = mapped_column(Boolean, default=False) # Pre-defined vs custom
  18. is_deleted: Mapped[bool] = mapped_column(Boolean, default=False) # Hidden/removed type
  19. created_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now())
  20. # Relationships
  21. printer_maintenance: Mapped[list["PrinterMaintenance"]] = relationship(
  22. back_populates="maintenance_type", cascade="all, delete-orphan"
  23. )
  24. class PrinterMaintenance(Base):
  25. """Tracks maintenance status for a specific printer."""
  26. __tablename__ = "printer_maintenance"
  27. id: Mapped[int] = mapped_column(primary_key=True)
  28. printer_id: Mapped[int] = mapped_column(ForeignKey("printers.id", ondelete="CASCADE"))
  29. maintenance_type_id: Mapped[int] = mapped_column(ForeignKey("maintenance_types.id", ondelete="CASCADE"))
  30. # Custom interval for this printer (overrides default if set)
  31. custom_interval_hours: Mapped[float | None] = mapped_column(Float, nullable=True)
  32. # Custom interval type for this printer (overrides default if set)
  33. custom_interval_type: Mapped[str | None] = mapped_column(String(20), nullable=True)
  34. # Tracking
  35. enabled: Mapped[bool] = mapped_column(Boolean, default=True)
  36. last_performed_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True)
  37. last_performed_hours: Mapped[float] = mapped_column(Float, default=0.0) # Hours at last reset
  38. created_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now())
  39. updated_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now(), onupdate=func.now())
  40. # Relationships
  41. printer: Mapped["Printer"] = relationship(back_populates="maintenance_items")
  42. maintenance_type: Mapped["MaintenanceType"] = relationship(back_populates="printer_maintenance")
  43. history: Mapped[list["MaintenanceHistory"]] = relationship(
  44. back_populates="printer_maintenance", cascade="all, delete-orphan"
  45. )
  46. class MaintenanceHistory(Base):
  47. """Log of maintenance actions performed."""
  48. __tablename__ = "maintenance_history"
  49. id: Mapped[int] = mapped_column(primary_key=True)
  50. printer_maintenance_id: Mapped[int] = mapped_column(ForeignKey("printer_maintenance.id", ondelete="CASCADE"))
  51. performed_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now())
  52. hours_at_maintenance: Mapped[float] = mapped_column(Float, default=0.0)
  53. notes: Mapped[str | None] = mapped_column(Text, nullable=True)
  54. # Relationships
  55. printer_maintenance: Mapped["PrinterMaintenance"] = relationship(back_populates="history")
  56. # Import at end to avoid circular imports
  57. from backend.app.models.printer import Printer # noqa: E402