project_bom.py 1.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950
  1. from datetime import datetime
  2. from sqlalchemy import 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 ProjectBOMItem(Base):
  6. """Bill of Materials item for a project.
  7. Tracks sourced/purchased parts (hardware, electronics, screws, etc.)
  8. that need to be acquired for a project.
  9. """
  10. __tablename__ = "project_bom_items"
  11. id: Mapped[int] = mapped_column(primary_key=True)
  12. project_id: Mapped[int] = mapped_column(ForeignKey("projects.id", ondelete="CASCADE"))
  13. name: Mapped[str] = mapped_column(String(255))
  14. quantity_needed: Mapped[int] = mapped_column(Integer, default=1)
  15. quantity_acquired: Mapped[int] = mapped_column(Integer, default=0)
  16. # Sourcing information
  17. unit_price: Mapped[float | None] = mapped_column(Float, nullable=True)
  18. sourcing_url: Mapped[str | None] = mapped_column(String(512), nullable=True)
  19. # Optional link to archive (for reference)
  20. archive_id: Mapped[int | None] = mapped_column(ForeignKey("print_archives.id", ondelete="SET NULL"), nullable=True)
  21. # Reference to attachment filename
  22. stl_filename: Mapped[str | None] = mapped_column(String(255), nullable=True)
  23. # Remarks about this part
  24. remarks: Mapped[str | None] = mapped_column(Text, nullable=True)
  25. # Sort order
  26. sort_order: Mapped[int] = mapped_column(Integer, default=0)
  27. # Timestamps
  28. created_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now())
  29. updated_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now(), onupdate=func.now())
  30. # Relationships
  31. project: Mapped["Project"] = relationship(back_populates="bom_items")
  32. archive: Mapped["PrintArchive | None"] = relationship()
  33. from backend.app.models.archive import PrintArchive # noqa: E402
  34. from backend.app.models.project import Project # noqa: E402