library.py 3.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586
  1. """Library models for file manager functionality."""
  2. from datetime import datetime
  3. from sqlalchemy import JSON, DateTime, ForeignKey, Integer, String, Text, func
  4. from sqlalchemy.orm import Mapped, mapped_column, relationship
  5. from backend.app.core.database import Base
  6. class LibraryFolder(Base):
  7. """Folder for organizing library files."""
  8. __tablename__ = "library_folders"
  9. id: Mapped[int] = mapped_column(primary_key=True)
  10. name: Mapped[str] = mapped_column(String(255))
  11. parent_id: Mapped[int | None] = mapped_column(ForeignKey("library_folders.id", ondelete="CASCADE"), nullable=True)
  12. # Link to project or archive
  13. project_id: Mapped[int | None] = mapped_column(ForeignKey("projects.id", ondelete="SET NULL"), nullable=True)
  14. archive_id: Mapped[int | None] = mapped_column(ForeignKey("print_archives.id", ondelete="SET NULL"), nullable=True)
  15. # Timestamps
  16. created_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now())
  17. updated_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now(), onupdate=func.now())
  18. # Relationships
  19. parent: Mapped["LibraryFolder | None"] = relationship(
  20. "LibraryFolder",
  21. back_populates="children",
  22. remote_side="LibraryFolder.id",
  23. foreign_keys="LibraryFolder.parent_id",
  24. )
  25. children: Mapped[list["LibraryFolder"]] = relationship(
  26. "LibraryFolder",
  27. back_populates="parent",
  28. foreign_keys="LibraryFolder.parent_id",
  29. cascade="all, delete-orphan",
  30. )
  31. files: Mapped[list["LibraryFile"]] = relationship(
  32. back_populates="folder",
  33. cascade="all, delete-orphan",
  34. )
  35. project: Mapped["Project | None"] = relationship()
  36. archive: Mapped["PrintArchive | None"] = relationship()
  37. class LibraryFile(Base):
  38. """File stored in the library."""
  39. __tablename__ = "library_files"
  40. id: Mapped[int] = mapped_column(primary_key=True)
  41. folder_id: Mapped[int | None] = mapped_column(ForeignKey("library_folders.id", ondelete="CASCADE"), nullable=True)
  42. project_id: Mapped[int | None] = mapped_column(ForeignKey("projects.id", ondelete="SET NULL"), nullable=True)
  43. # File info
  44. filename: Mapped[str] = mapped_column(String(255)) # Original filename
  45. file_path: Mapped[str] = mapped_column(String(500)) # Storage path
  46. file_type: Mapped[str] = mapped_column(String(10)) # "3mf" or "gcode"
  47. file_size: Mapped[int] = mapped_column(Integer)
  48. file_hash: Mapped[str | None] = mapped_column(String(64)) # SHA256 for duplicate detection
  49. thumbnail_path: Mapped[str | None] = mapped_column(String(500))
  50. # Extracted metadata (from 3MF parser)
  51. file_metadata: Mapped[dict | None] = mapped_column(JSON)
  52. # Usage tracking
  53. print_count: Mapped[int] = mapped_column(Integer, default=0)
  54. last_printed_at: Mapped[datetime | None] = mapped_column(DateTime, nullable=True)
  55. # User notes
  56. notes: Mapped[str | None] = mapped_column(Text, nullable=True)
  57. # Timestamps
  58. created_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now())
  59. updated_at: Mapped[datetime] = mapped_column(DateTime, server_default=func.now(), onupdate=func.now())
  60. # Relationships
  61. folder: Mapped["LibraryFolder | None"] = relationship(back_populates="files")
  62. project: Mapped["Project | None"] = relationship()
  63. from backend.app.models.archive import PrintArchive # noqa: E402, F811
  64. from backend.app.models.project import Project # noqa: E402, F811