project.py 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. from datetime import datetime
  2. from pydantic import BaseModel
  3. class ProjectCreate(BaseModel):
  4. """Schema for creating a new project."""
  5. name: str
  6. description: str | None = None
  7. color: str | None = None
  8. target_count: int | None = None
  9. notes: str | None = None
  10. tags: str | None = None
  11. due_date: datetime | None = None
  12. priority: str = "normal"
  13. budget: float | None = None
  14. parent_id: int | None = None # For sub-projects
  15. class ProjectUpdate(BaseModel):
  16. """Schema for updating a project."""
  17. name: str | None = None
  18. description: str | None = None
  19. color: str | None = None
  20. status: str | None = None # active, completed, archived
  21. target_count: int | None = None
  22. notes: str | None = None
  23. tags: str | None = None
  24. due_date: datetime | None = None
  25. priority: str | None = None
  26. budget: float | None = None
  27. parent_id: int | None = None
  28. class ProjectStats(BaseModel):
  29. """Statistics for a project."""
  30. total_archives: int = 0
  31. completed_prints: int = 0
  32. failed_prints: int = 0
  33. queued_prints: int = 0
  34. in_progress_prints: int = 0
  35. total_print_time_hours: float = 0.0
  36. total_filament_grams: float = 0.0
  37. progress_percent: float | None = None # Based on target_count
  38. # Cost tracking (Phase 6)
  39. estimated_cost: float = 0.0 # Based on filament cost
  40. total_energy_kwh: float = 0.0
  41. total_energy_cost: float = 0.0
  42. remaining_prints: int | None = None # target_count - completed_prints
  43. # BOM stats (Phase 7)
  44. bom_total_items: int = 0
  45. bom_completed_items: int = 0
  46. class ProjectChildPreview(BaseModel):
  47. """Minimal project data for child preview."""
  48. id: int
  49. name: str
  50. color: str | None
  51. status: str
  52. progress_percent: float | None = None
  53. class ProjectResponse(BaseModel):
  54. """Schema for project response."""
  55. id: int
  56. name: str
  57. description: str | None
  58. color: str | None
  59. status: str
  60. target_count: int | None
  61. notes: str | None = None
  62. attachments: list | None = None
  63. tags: str | None = None
  64. due_date: datetime | None = None
  65. priority: str = "normal"
  66. budget: float | None = None
  67. is_template: bool = False
  68. template_source_id: int | None = None
  69. parent_id: int | None = None
  70. parent_name: str | None = None # For display
  71. children: list[ProjectChildPreview] = []
  72. created_at: datetime
  73. updated_at: datetime
  74. stats: ProjectStats | None = None
  75. class Config:
  76. from_attributes = True
  77. class ArchivePreview(BaseModel):
  78. """Minimal archive data for project preview."""
  79. id: int
  80. print_name: str | None
  81. thumbnail_path: str | None
  82. status: str
  83. class ProjectListResponse(BaseModel):
  84. """Schema for project list item (lighter weight)."""
  85. id: int
  86. name: str
  87. description: str | None
  88. color: str | None
  89. status: str
  90. target_count: int | None
  91. created_at: datetime
  92. # Quick stats
  93. archive_count: int = 0
  94. queue_count: int = 0
  95. progress_percent: float | None = None
  96. # Preview of archives (up to 5)
  97. archives: list[ArchivePreview] = []
  98. class Config:
  99. from_attributes = True
  100. class BatchAddArchives(BaseModel):
  101. """Schema for batch adding archives to a project."""
  102. archive_ids: list[int]
  103. class BatchAddQueueItems(BaseModel):
  104. """Schema for batch adding queue items to a project."""
  105. queue_item_ids: list[int]
  106. # Phase 7: BOM Schemas
  107. class BOMItemCreate(BaseModel):
  108. """Schema for creating a BOM item."""
  109. name: str
  110. quantity_needed: int = 1
  111. archive_id: int | None = None
  112. stl_filename: str | None = None
  113. notes: str | None = None
  114. class BOMItemUpdate(BaseModel):
  115. """Schema for updating a BOM item."""
  116. name: str | None = None
  117. quantity_needed: int | None = None
  118. quantity_printed: int | None = None
  119. archive_id: int | None = None
  120. stl_filename: str | None = None
  121. notes: str | None = None
  122. class BOMItemResponse(BaseModel):
  123. """Schema for BOM item response."""
  124. id: int
  125. project_id: int
  126. name: str
  127. quantity_needed: int
  128. quantity_printed: int
  129. archive_id: int | None
  130. archive_name: str | None = None
  131. stl_filename: str | None
  132. notes: str | None
  133. sort_order: int
  134. is_complete: bool = False
  135. created_at: datetime
  136. updated_at: datetime
  137. class Config:
  138. from_attributes = True
  139. # Phase 9: Timeline Schemas
  140. class TimelineEvent(BaseModel):
  141. """Schema for a timeline event."""
  142. event_type: str # archive_added, queue_started, queue_completed, status_changed, note_updated
  143. timestamp: datetime
  144. title: str
  145. description: str | None = None
  146. metadata: dict | None = None # Additional event-specific data