archive.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. from datetime import datetime
  2. from sqlalchemy import JSON, Boolean, DateTime, Float, ForeignKey, Integer, String, Text, func
  3. from sqlalchemy.orm import Mapped, mapped_column, relationship
  4. from backend.app.core.database import Base
  5. class PrintArchive(Base):
  6. __tablename__ = "print_archives"
  7. id: Mapped[int] = mapped_column(primary_key=True)
  8. printer_id: Mapped[int | None] = mapped_column(ForeignKey("printers.id"), nullable=True)
  9. project_id: Mapped[int | None] = mapped_column(ForeignKey("projects.id", ondelete="SET NULL"), nullable=True)
  10. # File info
  11. filename: Mapped[str] = mapped_column(String(255))
  12. file_path: Mapped[str] = mapped_column(String(500))
  13. file_size: Mapped[int] = mapped_column(Integer)
  14. content_hash: Mapped[str | None] = mapped_column(String(64)) # SHA256 hash for duplicate detection
  15. thumbnail_path: Mapped[str | None] = mapped_column(String(500))
  16. timelapse_path: Mapped[str | None] = mapped_column(String(500))
  17. source_3mf_path: Mapped[str | None] = mapped_column(String(500)) # Original project 3MF from slicer
  18. f3d_path: Mapped[str | None] = mapped_column(String(500)) # Fusion 360 design file
  19. # Print details from 3MF / printer
  20. print_name: Mapped[str | None] = mapped_column(String(255))
  21. print_time_seconds: Mapped[int | None] = mapped_column(Integer)
  22. filament_used_grams: Mapped[float | None] = mapped_column(Float)
  23. filament_type: Mapped[str | None] = mapped_column(String(50))
  24. filament_color: Mapped[str | None] = mapped_column(String(200))
  25. layer_height: Mapped[float | None] = mapped_column(Float)
  26. total_layers: Mapped[int | None] = mapped_column(Integer)
  27. nozzle_diameter: Mapped[float | None] = mapped_column(Float)
  28. bed_temperature: Mapped[int | None] = mapped_column(Integer)
  29. bed_type: Mapped[str | None] = mapped_column(String(64)) # e.g. "Cool Plate", "Textured PEI Plate"
  30. nozzle_temperature: Mapped[int | None] = mapped_column(Integer)
  31. # Printer model this file was sliced for (extracted from 3MF metadata)
  32. sliced_for_model: Mapped[str | None] = mapped_column(String(50), nullable=True)
  33. # Print result
  34. status: Mapped[str] = mapped_column(String(20), default="completed")
  35. started_at: Mapped[datetime | None] = mapped_column(DateTime)
  36. completed_at: Mapped[datetime | None] = mapped_column(DateTime)
  37. # Printer-assigned subtask identifier from MQTT. Used to resume the same
  38. # archive row across a backend restart during a long-running print (#972):
  39. # if the same subtask_id reappears after restart, we know it's the same
  40. # print and keep the original row instead of cancel-then-create.
  41. subtask_id: Mapped[str | None] = mapped_column(String(64), nullable=True)
  42. # Extended metadata (JSON blob for flexibility)
  43. extra_data: Mapped[dict | None] = mapped_column(JSON)
  44. # MakerWorld info (auto-extracted from 3MF)
  45. makerworld_url: Mapped[str | None] = mapped_column(String(500))
  46. designer: Mapped[str | None] = mapped_column(String(255))
  47. # User-defined external link (Printables, Thingiverse, etc.)
  48. external_url: Mapped[str | None] = mapped_column(String(500))
  49. # User additions
  50. is_favorite: Mapped[bool] = mapped_column(Boolean, default=False)
  51. tags: Mapped[str | None] = mapped_column(Text)
  52. notes: Mapped[str | None] = mapped_column(Text)
  53. cost: Mapped[float | None] = mapped_column(Float)
  54. photos: Mapped[list | None] = mapped_column(JSON) # List of photo filenames
  55. failure_reason: Mapped[str | None] = mapped_column(String(100)) # For failed prints
  56. quantity: Mapped[int] = mapped_column(Integer, default=1) # Number of items printed
  57. # Energy tracking
  58. energy_kwh: Mapped[float | None] = mapped_column(Float) # Energy consumed in kWh
  59. energy_cost: Mapped[float | None] = mapped_column(Float) # Cost of energy consumed
  60. # Plug lifetime counter captured at print start; delta at print end becomes energy_kwh.
  61. # Persisted so per-print tracking survives backend restarts mid-print (#941).
  62. energy_start_kwh: Mapped[float | None] = mapped_column(Float)
  63. # Timestamps
  64. created_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now())
  65. # User tracking (who uploaded/created this archive)
  66. created_by_id: Mapped[int | None] = mapped_column(ForeignKey("users.id", ondelete="SET NULL"), nullable=True)
  67. # Relationships
  68. printer: Mapped["Printer | None"] = relationship(back_populates="archives")
  69. project: Mapped["Project | None"] = relationship(back_populates="archives")
  70. created_by: Mapped["User | None"] = relationship()
  71. from backend.app.models.printer import Printer # noqa: E402, F811
  72. from backend.app.models.project import Project # noqa: E402, F811
  73. from backend.app.models.user import User # noqa: E402, F811