active_print_spoolman.py 1.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142
  1. """Track Spoolman data for active prints."""
  2. from sqlalchemy import JSON, ForeignKey, UniqueConstraint
  3. from sqlalchemy.orm import Mapped, mapped_column
  4. from backend.app.core.database import Base
  5. class ActivePrintSpoolman(Base):
  6. """Stores Spoolman tracking data for active prints.
  7. This data is captured at print start and used at print completion
  8. to report per-filament usage to the correct Spoolman spools.
  9. Rows are deleted after print completes.
  10. Key: (printer_id, archive_id) - allows same archive on different printers
  11. """
  12. __tablename__ = "active_print_spoolman"
  13. __table_args__ = (UniqueConstraint("printer_id", "archive_id", name="uq_printer_archive"),)
  14. id: Mapped[int] = mapped_column(primary_key=True)
  15. printer_id: Mapped[int] = mapped_column(ForeignKey("printers.id", ondelete="CASCADE"))
  16. archive_id: Mapped[int] = mapped_column(ForeignKey("print_archives.id", ondelete="CASCADE"))
  17. # Per-filament usage from 3MF: [{"slot_id": 1, "used_g": 50.5, "type": "PLA"}, ...]
  18. filament_usage: Mapped[list] = mapped_column(JSON)
  19. # AMS tray state at print start: {0: {"tray_uuid": "...", "tag_uid": "..."}, ...}
  20. ams_trays: Mapped[dict] = mapped_column(JSON)
  21. # Custom slot-to-tray mapping from queue (optional): [5, -1, 2, -1]
  22. slot_to_tray: Mapped[list | None] = mapped_column(JSON, nullable=True)
  23. # Per-layer cumulative usage from G-code parsing (for accurate partial usage)
  24. # Format: {"0": {0: 125.5}, "1": {0: 250.0, 1: 50.0}, ...}
  25. # Keys are layer numbers (as strings for JSON), values are filament_id -> mm
  26. layer_usage: Mapped[dict | None] = mapped_column(JSON, nullable=True)
  27. # Filament properties (density, diameter per filament slot)
  28. # Format: {1: {"density": 1.24, "diameter": 1.75, "type": "PLA"}, ...}
  29. filament_properties: Mapped[dict | None] = mapped_column(JSON, nullable=True)