firmware.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. """
  2. Firmware Update API Routes
  3. Check for firmware updates from Bambu Lab.
  4. Also provides endpoints for uploading firmware to printers via SD card.
  5. """
  6. import logging
  7. from fastapi import APIRouter, Depends, HTTPException
  8. from pydantic import BaseModel, Field
  9. from sqlalchemy import select
  10. from sqlalchemy.ext.asyncio import AsyncSession
  11. from backend.app.core.auth import RequirePermissionIfAuthEnabled
  12. from backend.app.core.database import get_db
  13. from backend.app.core.permissions import Permission
  14. from backend.app.models.printer import Printer
  15. from backend.app.models.user import User
  16. from backend.app.services.firmware_check import get_firmware_service
  17. from backend.app.services.firmware_update import (
  18. FirmwareUploadStatus,
  19. get_firmware_update_service,
  20. get_upload_state,
  21. )
  22. from backend.app.services.printer_manager import printer_manager
  23. logger = logging.getLogger(__name__)
  24. router = APIRouter(prefix="/firmware", tags=["firmware"])
  25. class FirmwareUpdateInfo(BaseModel):
  26. """Firmware update information for a printer."""
  27. printer_id: int
  28. printer_name: str
  29. model: str | None
  30. current_version: str | None
  31. latest_version: str | None
  32. update_available: bool
  33. download_url: str | None = None
  34. release_notes: str | None = None
  35. class FirmwareUpdatesResponse(BaseModel):
  36. """Response containing firmware updates for all printers."""
  37. updates: list[FirmwareUpdateInfo] = Field(default_factory=list)
  38. updates_available: int = Field(0, description="Number of printers with updates available")
  39. class LatestFirmwareInfo(BaseModel):
  40. """Latest firmware version info for a model."""
  41. model_key: str
  42. version: str
  43. download_url: str
  44. release_notes: str | None = None
  45. @router.get("/updates", response_model=FirmwareUpdatesResponse)
  46. async def check_firmware_updates(
  47. db: AsyncSession = Depends(get_db),
  48. _: User | None = RequirePermissionIfAuthEnabled(Permission.FIRMWARE_READ),
  49. ):
  50. """
  51. Check for firmware updates for all connected printers.
  52. Compares each printer's current firmware version against the latest
  53. available version from Bambu Lab's official firmware download page.
  54. Note: This does not require cloud authentication - it uses public
  55. firmware information from bambulab.com.
  56. """
  57. firmware_service = get_firmware_service()
  58. # Get all printers from database
  59. result = await db.execute(select(Printer).where(Printer.is_active.is_(True)))
  60. printers = result.scalars().all()
  61. updates = []
  62. updates_available = 0
  63. for printer in printers:
  64. # Get current firmware version from MQTT state
  65. current_version = None
  66. mqtt_client = printer_manager.get_client(printer.id)
  67. if mqtt_client and mqtt_client.state:
  68. current_version = mqtt_client.state.firmware_version
  69. # Check for update
  70. model = printer.model or "Unknown"
  71. update_info = await firmware_service.check_for_update(model, current_version or "")
  72. if update_info["update_available"]:
  73. updates_available += 1
  74. updates.append(
  75. FirmwareUpdateInfo(
  76. printer_id=printer.id,
  77. printer_name=printer.name,
  78. model=model,
  79. current_version=current_version,
  80. latest_version=update_info["latest_version"],
  81. update_available=update_info["update_available"],
  82. download_url=update_info["download_url"],
  83. release_notes=update_info["release_notes"],
  84. )
  85. )
  86. return FirmwareUpdatesResponse(updates=updates, updates_available=updates_available)
  87. @router.get("/updates/{printer_id}", response_model=FirmwareUpdateInfo)
  88. async def check_printer_firmware(
  89. printer_id: int,
  90. db: AsyncSession = Depends(get_db),
  91. _: User | None = RequirePermissionIfAuthEnabled(Permission.FIRMWARE_READ),
  92. ):
  93. """
  94. Check for firmware update for a specific printer.
  95. """
  96. firmware_service = get_firmware_service()
  97. # Get printer from database
  98. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  99. printer = result.scalar_one_or_none()
  100. if not printer:
  101. raise HTTPException(status_code=404, detail="Printer not found")
  102. # Get current firmware version from MQTT state
  103. current_version = None
  104. mqtt_client = printer_manager.get_client(printer.id)
  105. if mqtt_client and mqtt_client.state:
  106. current_version = mqtt_client.state.firmware_version
  107. # Check for update
  108. model = printer.model or "Unknown"
  109. update_info = await firmware_service.check_for_update(model, current_version or "")
  110. return FirmwareUpdateInfo(
  111. printer_id=printer.id,
  112. printer_name=printer.name,
  113. model=model,
  114. current_version=current_version,
  115. latest_version=update_info["latest_version"],
  116. update_available=update_info["update_available"],
  117. download_url=update_info["download_url"],
  118. release_notes=update_info["release_notes"],
  119. )
  120. @router.get("/latest", response_model=list[LatestFirmwareInfo])
  121. async def get_all_latest_firmware(
  122. _: User | None = RequirePermissionIfAuthEnabled(Permission.FIRMWARE_READ),
  123. ):
  124. """
  125. Get the latest firmware versions for all Bambu Lab printer models.
  126. This endpoint fetches the latest available firmware versions from
  127. Bambu Lab's official firmware download page.
  128. """
  129. firmware_service = get_firmware_service()
  130. versions = await firmware_service.get_all_latest_versions()
  131. return [
  132. LatestFirmwareInfo(
  133. model_key=key,
  134. version=info.version,
  135. download_url=info.download_url,
  136. release_notes=info.release_notes,
  137. )
  138. for key, info in versions.items()
  139. ]
  140. # ============================================================================
  141. # Firmware Upload Endpoints (for LAN-only firmware updates)
  142. # ============================================================================
  143. class FirmwareUploadPrepareResponse(BaseModel):
  144. """Response from firmware upload preparation check."""
  145. can_proceed: bool
  146. sd_card_present: bool
  147. sd_card_free_space: int = Field(-1, description="Free space in bytes, -1 if unknown")
  148. firmware_size: int = Field(0, description="Estimated firmware size in bytes")
  149. space_sufficient: bool
  150. update_available: bool
  151. current_version: str | None = None
  152. latest_version: str | None = None
  153. firmware_filename: str | None = None
  154. errors: list[str] = Field(default_factory=list)
  155. class FirmwareUploadStatusResponse(BaseModel):
  156. """Response containing firmware upload status."""
  157. status: str # idle, preparing, downloading, uploading, complete, error
  158. progress: int = Field(0, ge=0, le=100)
  159. message: str = ""
  160. error: str | None = None
  161. firmware_filename: str | None = None
  162. firmware_version: str | None = None
  163. class FirmwareUploadStartResponse(BaseModel):
  164. """Response when starting a firmware upload."""
  165. started: bool
  166. message: str
  167. @router.get("/updates/{printer_id}/prepare", response_model=FirmwareUploadPrepareResponse)
  168. async def prepare_firmware_upload(
  169. printer_id: int,
  170. db: AsyncSession = Depends(get_db),
  171. _: User | None = RequirePermissionIfAuthEnabled(Permission.FIRMWARE_READ),
  172. ):
  173. """
  174. Check prerequisites for uploading firmware to a printer.
  175. This performs pre-flight checks including:
  176. - SD card presence
  177. - Available storage space
  178. - Update availability
  179. Call this before starting a firmware upload to ensure the operation
  180. can succeed.
  181. """
  182. update_service = get_firmware_update_service()
  183. result = await update_service.prepare_update(printer_id, db)
  184. return FirmwareUploadPrepareResponse(**result)
  185. @router.post("/updates/{printer_id}/upload", response_model=FirmwareUploadStartResponse)
  186. async def start_firmware_upload(
  187. printer_id: int,
  188. db: AsyncSession = Depends(get_db),
  189. _: User | None = RequirePermissionIfAuthEnabled(Permission.FIRMWARE_UPDATE),
  190. ):
  191. """
  192. Start uploading firmware to a printer's SD card.
  193. This initiates a background process that:
  194. 1. Downloads the firmware from Bambu Lab
  195. 2. Uploads it to the printer's SD card via FTP
  196. Progress is broadcast via WebSocket with type "firmware_upload_progress".
  197. Use GET /firmware/updates/{printer_id}/upload/status for polling fallback.
  198. After upload completes, the user must trigger the update from the
  199. printer's screen (Settings > Firmware).
  200. """
  201. # First check prerequisites
  202. update_service = get_firmware_update_service()
  203. prepare_result = await update_service.prepare_update(printer_id, db)
  204. if not prepare_result["can_proceed"]:
  205. errors = prepare_result.get("errors", ["Cannot proceed with firmware upload"])
  206. raise HTTPException(
  207. status_code=400,
  208. detail="; ".join(errors),
  209. )
  210. # Start the upload
  211. started = await update_service.start_upload(printer_id, db)
  212. if not started:
  213. state = get_upload_state(printer_id)
  214. if state.status == FirmwareUploadStatus.DOWNLOADING:
  215. return FirmwareUploadStartResponse(
  216. started=False,
  217. message="Firmware upload already in progress",
  218. )
  219. raise HTTPException(
  220. status_code=500,
  221. detail=state.error or "Failed to start firmware upload",
  222. )
  223. return FirmwareUploadStartResponse(
  224. started=True,
  225. message="Firmware upload started. Progress will be broadcast via WebSocket.",
  226. )
  227. @router.get("/updates/{printer_id}/upload/status", response_model=FirmwareUploadStatusResponse)
  228. async def get_firmware_upload_status(
  229. printer_id: int,
  230. _: User | None = RequirePermissionIfAuthEnabled(Permission.FIRMWARE_READ),
  231. ):
  232. """
  233. Get the current status of a firmware upload operation.
  234. This is a polling fallback for clients that don't use WebSocket.
  235. For real-time updates, connect to WebSocket and listen for
  236. "firmware_upload_progress" messages.
  237. """
  238. state = get_upload_state(printer_id)
  239. return FirmwareUploadStatusResponse(
  240. status=state.status.value,
  241. progress=state.progress,
  242. message=state.message,
  243. error=state.error,
  244. firmware_filename=state.firmware_filename,
  245. firmware_version=state.firmware_version,
  246. )