project.py 5.7 KB

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