printers.py 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899
  1. import io
  2. import logging
  3. import zipfile
  4. from pathlib import Path
  5. from fastapi import APIRouter, Depends, HTTPException
  6. logger = logging.getLogger(__name__)
  7. from fastapi.responses import Response
  8. from sqlalchemy.ext.asyncio import AsyncSession
  9. from sqlalchemy import select
  10. from backend.app.core.database import get_db
  11. from backend.app.core.config import settings
  12. from backend.app.models.printer import Printer
  13. from backend.app.models.slot_preset import SlotPresetMapping
  14. from backend.app.schemas.printer import (
  15. PrinterCreate,
  16. PrinterUpdate,
  17. PrinterResponse,
  18. PrinterStatus,
  19. HMSErrorResponse,
  20. AMSUnit,
  21. AMSTray,
  22. NozzleInfoResponse,
  23. PrintOptionsResponse,
  24. )
  25. from backend.app.services.printer_manager import printer_manager
  26. from backend.app.services.bambu_mqtt import get_stage_name
  27. from backend.app.services.bambu_ftp import (
  28. download_file_try_paths_async,
  29. list_files_async,
  30. delete_file_async,
  31. download_file_bytes_async,
  32. get_storage_info_async,
  33. )
  34. router = APIRouter(prefix="/printers", tags=["printers"])
  35. @router.get("/", response_model=list[PrinterResponse])
  36. async def list_printers(db: AsyncSession = Depends(get_db)):
  37. """List all configured printers."""
  38. result = await db.execute(select(Printer).order_by(Printer.name))
  39. return list(result.scalars().all())
  40. @router.post("/", response_model=PrinterResponse)
  41. async def create_printer(
  42. printer_data: PrinterCreate,
  43. db: AsyncSession = Depends(get_db),
  44. ):
  45. """Add a new printer."""
  46. # Check if serial number already exists
  47. result = await db.execute(
  48. select(Printer).where(Printer.serial_number == printer_data.serial_number)
  49. )
  50. if result.scalar_one_or_none():
  51. raise HTTPException(400, "Printer with this serial number already exists")
  52. printer = Printer(**printer_data.model_dump())
  53. db.add(printer)
  54. await db.commit()
  55. await db.refresh(printer)
  56. # Connect to the printer
  57. if printer.is_active:
  58. await printer_manager.connect_printer(printer)
  59. return printer
  60. @router.get("/{printer_id}", response_model=PrinterResponse)
  61. async def get_printer(printer_id: int, db: AsyncSession = Depends(get_db)):
  62. """Get a specific printer."""
  63. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  64. printer = result.scalar_one_or_none()
  65. if not printer:
  66. raise HTTPException(404, "Printer not found")
  67. return printer
  68. @router.patch("/{printer_id}", response_model=PrinterResponse)
  69. async def update_printer(
  70. printer_id: int,
  71. printer_data: PrinterUpdate,
  72. db: AsyncSession = Depends(get_db),
  73. ):
  74. """Update a printer."""
  75. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  76. printer = result.scalar_one_or_none()
  77. if not printer:
  78. raise HTTPException(404, "Printer not found")
  79. update_data = printer_data.model_dump(exclude_unset=True)
  80. for field, value in update_data.items():
  81. setattr(printer, field, value)
  82. await db.commit()
  83. await db.refresh(printer)
  84. # Reconnect if connection settings changed
  85. if any(k in update_data for k in ["ip_address", "access_code", "is_active"]):
  86. printer_manager.disconnect_printer(printer_id)
  87. if printer.is_active:
  88. await printer_manager.connect_printer(printer)
  89. return printer
  90. @router.delete("/{printer_id}")
  91. async def delete_printer(printer_id: int, db: AsyncSession = Depends(get_db)):
  92. """Delete a printer."""
  93. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  94. printer = result.scalar_one_or_none()
  95. if not printer:
  96. raise HTTPException(404, "Printer not found")
  97. printer_manager.disconnect_printer(printer_id)
  98. await db.delete(printer)
  99. await db.commit()
  100. return {"status": "deleted"}
  101. @router.get("/{printer_id}/status", response_model=PrinterStatus)
  102. async def get_printer_status(printer_id: int, db: AsyncSession = Depends(get_db)):
  103. """Get real-time status of a printer."""
  104. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  105. printer = result.scalar_one_or_none()
  106. if not printer:
  107. raise HTTPException(404, "Printer not found")
  108. state = printer_manager.get_status(printer_id)
  109. if not state:
  110. return PrinterStatus(
  111. id=printer_id,
  112. name=printer.name,
  113. connected=False,
  114. )
  115. # Determine cover URL if there's an active print
  116. cover_url = None
  117. if state.state == "RUNNING" and state.gcode_file:
  118. cover_url = f"/api/v1/printers/{printer_id}/cover"
  119. # Convert HMS errors to response format
  120. hms_errors = [
  121. HMSErrorResponse(code=e.code, attr=e.attr, module=e.module, severity=e.severity)
  122. for e in (state.hms_errors or [])
  123. ]
  124. # Parse AMS data from raw_data
  125. ams_units = []
  126. vt_tray = None
  127. ams_exists = False
  128. raw_data = state.raw_data or {}
  129. if "ams" in raw_data and isinstance(raw_data["ams"], list):
  130. ams_exists = True
  131. for ams_data in raw_data["ams"]:
  132. # Skip if ams_data is not a dict (defensive check)
  133. if not isinstance(ams_data, dict):
  134. continue
  135. trays = []
  136. for tray_data in ams_data.get("tray", []):
  137. # Filter out empty/invalid tag values
  138. tag_uid = tray_data.get("tag_uid", "")
  139. if tag_uid in ("", "0000000000000000"):
  140. tag_uid = None
  141. tray_uuid = tray_data.get("tray_uuid", "")
  142. if tray_uuid in ("", "00000000000000000000000000000000"):
  143. tray_uuid = None
  144. trays.append(AMSTray(
  145. id=tray_data.get("id", 0),
  146. tray_color=tray_data.get("tray_color"),
  147. tray_type=tray_data.get("tray_type"),
  148. tray_sub_brands=tray_data.get("tray_sub_brands"),
  149. tray_id_name=tray_data.get("tray_id_name"),
  150. tray_info_idx=tray_data.get("tray_info_idx"),
  151. remain=tray_data.get("remain", 0),
  152. k=tray_data.get("k"),
  153. tag_uid=tag_uid,
  154. tray_uuid=tray_uuid,
  155. nozzle_temp_min=tray_data.get("nozzle_temp_min"),
  156. nozzle_temp_max=tray_data.get("nozzle_temp_max"),
  157. ))
  158. # Prefer humidity_raw (percentage) over humidity (index 1-5)
  159. # humidity_raw is the actual percentage value from the sensor
  160. humidity_raw = ams_data.get("humidity_raw")
  161. humidity_idx = ams_data.get("humidity")
  162. # Use humidity_raw if available, otherwise fall back to humidity index
  163. humidity_value = None
  164. if humidity_raw is not None:
  165. try:
  166. humidity_value = int(humidity_raw)
  167. except (ValueError, TypeError):
  168. pass
  169. if humidity_value is None and humidity_idx is not None:
  170. try:
  171. humidity_value = int(humidity_idx)
  172. except (ValueError, TypeError):
  173. pass
  174. ams_units.append(AMSUnit(
  175. id=ams_data.get("id", 0),
  176. humidity=humidity_value,
  177. temp=ams_data.get("temp"),
  178. tray=trays,
  179. ))
  180. # Virtual tray (external spool holder) - comes from vt_tray in raw_data
  181. if "vt_tray" in raw_data:
  182. vt_data = raw_data["vt_tray"]
  183. # Filter out empty/invalid tag values for vt_tray
  184. vt_tag_uid = vt_data.get("tag_uid", "")
  185. if vt_tag_uid in ("", "0000000000000000"):
  186. vt_tag_uid = None
  187. vt_tray_uuid = vt_data.get("tray_uuid", "")
  188. if vt_tray_uuid in ("", "00000000000000000000000000000000"):
  189. vt_tray_uuid = None
  190. vt_tray = AMSTray(
  191. id=254, # Virtual tray ID
  192. tray_color=vt_data.get("tray_color"),
  193. tray_type=vt_data.get("tray_type"),
  194. tray_sub_brands=vt_data.get("tray_sub_brands"),
  195. remain=vt_data.get("remain", 0),
  196. k=vt_data.get("k"),
  197. tag_uid=vt_tag_uid,
  198. tray_uuid=vt_tray_uuid,
  199. nozzle_temp_min=vt_data.get("nozzle_temp_min"),
  200. nozzle_temp_max=vt_data.get("nozzle_temp_max"),
  201. )
  202. # Convert nozzle info to response format
  203. nozzles = [
  204. NozzleInfoResponse(
  205. nozzle_type=n.nozzle_type,
  206. nozzle_diameter=n.nozzle_diameter,
  207. )
  208. for n in (state.nozzles or [])
  209. ]
  210. # Convert print options to response format
  211. print_options = PrintOptionsResponse(
  212. spaghetti_detector=state.print_options.spaghetti_detector,
  213. print_halt=state.print_options.print_halt,
  214. halt_print_sensitivity=state.print_options.halt_print_sensitivity,
  215. first_layer_inspector=state.print_options.first_layer_inspector,
  216. printing_monitor=state.print_options.printing_monitor,
  217. buildplate_marker_detector=state.print_options.buildplate_marker_detector,
  218. allow_skip_parts=state.print_options.allow_skip_parts,
  219. nozzle_clumping_detector=state.print_options.nozzle_clumping_detector,
  220. nozzle_clumping_sensitivity=state.print_options.nozzle_clumping_sensitivity,
  221. pileup_detector=state.print_options.pileup_detector,
  222. pileup_sensitivity=state.print_options.pileup_sensitivity,
  223. airprint_detector=state.print_options.airprint_detector,
  224. airprint_sensitivity=state.print_options.airprint_sensitivity,
  225. auto_recovery_step_loss=state.print_options.auto_recovery_step_loss,
  226. filament_tangle_detect=state.print_options.filament_tangle_detect,
  227. )
  228. # Get AMS mapping from raw_data (which AMS is connected to which nozzle)
  229. ams_mapping = raw_data.get("ams_mapping", [])
  230. # Get per-AMS extruder map: {ams_id: extruder_id} where 0=right, 1=left
  231. ams_extruder_map = raw_data.get("ams_extruder_map", {})
  232. logger.debug(f"API returning ams_mapping: {ams_mapping}, ams_extruder_map: {ams_extruder_map}")
  233. # tray_now from MQTT is already a global tray ID: (ams_id * 4) + slot_id
  234. # Per OpenBambuAPI docs: 254 = external spool, 255 = no filament, otherwise global tray ID
  235. # No conversion needed - just use the raw value directly
  236. tray_now = state.tray_now
  237. logger.debug(f"Using tray_now directly as global ID: {tray_now}")
  238. return PrinterStatus(
  239. id=printer_id,
  240. name=printer.name,
  241. connected=state.connected,
  242. state=state.state,
  243. current_print=state.current_print,
  244. subtask_name=state.subtask_name,
  245. gcode_file=state.gcode_file,
  246. progress=state.progress,
  247. remaining_time=state.remaining_time,
  248. layer_num=state.layer_num,
  249. total_layers=state.total_layers,
  250. temperatures=state.temperatures,
  251. cover_url=cover_url,
  252. hms_errors=hms_errors,
  253. ams=ams_units,
  254. ams_exists=ams_exists,
  255. vt_tray=vt_tray,
  256. sdcard=state.sdcard,
  257. store_to_sdcard=state.store_to_sdcard,
  258. timelapse=state.timelapse,
  259. ipcam=state.ipcam,
  260. wifi_signal=state.wifi_signal,
  261. nozzles=nozzles,
  262. print_options=print_options,
  263. stg_cur=state.stg_cur,
  264. stg_cur_name=get_stage_name(state.stg_cur) if state.stg_cur >= 0 else None,
  265. stg=state.stg,
  266. airduct_mode=state.airduct_mode,
  267. speed_level=state.speed_level,
  268. chamber_light=state.chamber_light,
  269. active_extruder=state.active_extruder,
  270. ams_mapping=ams_mapping,
  271. ams_extruder_map=ams_extruder_map,
  272. tray_now=tray_now,
  273. ams_status_main=state.ams_status_main,
  274. ams_status_sub=state.ams_status_sub,
  275. mc_print_sub_stage=state.mc_print_sub_stage,
  276. last_ams_update=state.last_ams_update,
  277. )
  278. @router.post("/{printer_id}/connect")
  279. async def connect_printer(printer_id: int, db: AsyncSession = Depends(get_db)):
  280. """Manually connect to a printer."""
  281. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  282. printer = result.scalar_one_or_none()
  283. if not printer:
  284. raise HTTPException(404, "Printer not found")
  285. success = await printer_manager.connect_printer(printer)
  286. return {"connected": success}
  287. @router.post("/{printer_id}/disconnect")
  288. async def disconnect_printer(printer_id: int, db: AsyncSession = Depends(get_db)):
  289. """Manually disconnect from a printer."""
  290. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  291. printer = result.scalar_one_or_none()
  292. if not printer:
  293. raise HTTPException(404, "Printer not found")
  294. printer_manager.disconnect_printer(printer_id)
  295. return {"connected": False}
  296. @router.post("/test")
  297. async def test_printer_connection(
  298. ip_address: str,
  299. serial_number: str,
  300. access_code: str,
  301. ):
  302. """Test connection to a printer without saving."""
  303. result = await printer_manager.test_connection(
  304. ip_address=ip_address,
  305. serial_number=serial_number,
  306. access_code=access_code,
  307. )
  308. return result
  309. # Cache for cover images (printer_id -> (gcode_file, image_bytes))
  310. _cover_cache: dict[int, tuple[str, bytes]] = {}
  311. @router.get("/{printer_id}/cover")
  312. async def get_printer_cover(printer_id: int, db: AsyncSession = Depends(get_db)):
  313. """Get the cover image for the current print job."""
  314. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  315. printer = result.scalar_one_or_none()
  316. if not printer:
  317. raise HTTPException(404, "Printer not found")
  318. state = printer_manager.get_status(printer_id)
  319. if not state:
  320. raise HTTPException(404, "Printer not connected")
  321. # Use subtask_name as the 3MF filename (gcode_file is the path inside the 3MF)
  322. subtask_name = state.subtask_name
  323. if not subtask_name:
  324. raise HTTPException(404, f"No subtask_name in printer state (state={state.state})")
  325. # Check cache
  326. if printer_id in _cover_cache:
  327. cached_file, cached_image = _cover_cache[printer_id]
  328. if cached_file == subtask_name:
  329. return Response(content=cached_image, media_type="image/png")
  330. # Build 3MF filename from subtask_name
  331. # Bambu printers store files as "name.gcode.3mf"
  332. filename = subtask_name
  333. if not filename.endswith(".3mf"):
  334. filename = filename + ".gcode.3mf"
  335. # Try to download the 3MF file from printer
  336. temp_path = settings.archive_dir / "temp" / f"cover_{printer_id}_{filename}"
  337. temp_path.parent.mkdir(parents=True, exist_ok=True)
  338. remote_paths = [
  339. f"/{filename}", # Root directory (most common)
  340. f"/cache/{filename}",
  341. f"/model/{filename}",
  342. f"/data/{filename}",
  343. ]
  344. logger.info(f"Trying to download cover for '{filename}' from {printer.ip_address}")
  345. try:
  346. downloaded = await download_file_try_paths_async(
  347. printer.ip_address,
  348. printer.access_code,
  349. remote_paths,
  350. temp_path,
  351. )
  352. except Exception as e:
  353. logger.error(f"FTP download exception: {e}")
  354. raise HTTPException(500, f"FTP download failed: {e}")
  355. if not downloaded:
  356. raise HTTPException(404, f"Could not download 3MF file '{filename}' from printer {printer.ip_address}. Tried: {remote_paths}")
  357. # Verify file actually exists and has content
  358. if not temp_path.exists():
  359. raise HTTPException(500, f"Download reported success but file not found: {temp_path}")
  360. file_size = temp_path.stat().st_size
  361. logger.info(f"Downloaded file size: {file_size} bytes")
  362. if file_size == 0:
  363. temp_path.unlink()
  364. raise HTTPException(500, f"Downloaded file is empty: {filename}")
  365. try:
  366. # Extract thumbnail from 3MF (which is a ZIP file)
  367. try:
  368. zf = zipfile.ZipFile(temp_path, 'r')
  369. except zipfile.BadZipFile as e:
  370. raise HTTPException(500, f"Downloaded file is not a valid 3MF/ZIP: {e}")
  371. except Exception as e:
  372. raise HTTPException(500, f"Failed to open 3MF file: {e}")
  373. try:
  374. # Try common thumbnail paths in 3MF files
  375. thumbnail_paths = [
  376. "Metadata/plate_1.png",
  377. "Metadata/thumbnail.png",
  378. "Metadata/plate_1_small.png",
  379. "Thumbnails/thumbnail.png",
  380. "thumbnail.png",
  381. ]
  382. for thumb_path in thumbnail_paths:
  383. try:
  384. image_data = zf.read(thumb_path)
  385. # Cache the result
  386. _cover_cache[printer_id] = (subtask_name, image_data)
  387. return Response(content=image_data, media_type="image/png")
  388. except KeyError:
  389. continue
  390. # If no specific thumbnail found, try any PNG in Metadata
  391. for name in zf.namelist():
  392. if name.startswith("Metadata/") and name.endswith(".png"):
  393. image_data = zf.read(name)
  394. _cover_cache[printer_id] = (subtask_name, image_data)
  395. return Response(content=image_data, media_type="image/png")
  396. raise HTTPException(404, "No thumbnail found in 3MF file")
  397. finally:
  398. zf.close()
  399. finally:
  400. if temp_path.exists():
  401. temp_path.unlink()
  402. # ============================================
  403. # File Manager Endpoints
  404. # ============================================
  405. @router.get("/{printer_id}/files")
  406. async def list_printer_files(
  407. printer_id: int,
  408. path: str = "/",
  409. db: AsyncSession = Depends(get_db),
  410. ):
  411. """List files on the printer at the specified path."""
  412. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  413. printer = result.scalar_one_or_none()
  414. if not printer:
  415. raise HTTPException(404, "Printer not found")
  416. files = await list_files_async(printer.ip_address, printer.access_code, path)
  417. # Add full path to each file
  418. for f in files:
  419. f["path"] = f"{path.rstrip('/')}/{f['name']}" if path != "/" else f"/{f['name']}"
  420. return {
  421. "path": path,
  422. "files": files,
  423. }
  424. @router.get("/{printer_id}/files/download")
  425. async def download_printer_file(
  426. printer_id: int,
  427. path: str,
  428. db: AsyncSession = Depends(get_db),
  429. ):
  430. """Download a file from the printer."""
  431. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  432. printer = result.scalar_one_or_none()
  433. if not printer:
  434. raise HTTPException(404, "Printer not found")
  435. data = await download_file_bytes_async(printer.ip_address, printer.access_code, path)
  436. if data is None:
  437. raise HTTPException(404, f"File not found: {path}")
  438. # Determine content type based on extension
  439. filename = path.split("/")[-1]
  440. ext = filename.lower().split(".")[-1] if "." in filename else ""
  441. content_types = {
  442. "3mf": "application/vnd.ms-package.3dmanufacturing-3dmodel+xml",
  443. "gcode": "text/plain",
  444. "mp4": "video/mp4",
  445. "avi": "video/x-msvideo",
  446. "png": "image/png",
  447. "jpg": "image/jpeg",
  448. "jpeg": "image/jpeg",
  449. "json": "application/json",
  450. "txt": "text/plain",
  451. }
  452. content_type = content_types.get(ext, "application/octet-stream")
  453. return Response(
  454. content=data,
  455. media_type=content_type,
  456. headers={"Content-Disposition": f'attachment; filename="{filename}"'},
  457. )
  458. @router.delete("/{printer_id}/files")
  459. async def delete_printer_file(
  460. printer_id: int,
  461. path: str,
  462. db: AsyncSession = Depends(get_db),
  463. ):
  464. """Delete a file from the printer."""
  465. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  466. printer = result.scalar_one_or_none()
  467. if not printer:
  468. raise HTTPException(404, "Printer not found")
  469. success = await delete_file_async(printer.ip_address, printer.access_code, path)
  470. if not success:
  471. raise HTTPException(500, f"Failed to delete file: {path}")
  472. return {"status": "deleted", "path": path}
  473. @router.get("/{printer_id}/storage")
  474. async def get_printer_storage(
  475. printer_id: int,
  476. db: AsyncSession = Depends(get_db),
  477. ):
  478. """Get storage information from the printer."""
  479. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  480. printer = result.scalar_one_or_none()
  481. if not printer:
  482. raise HTTPException(404, "Printer not found")
  483. storage_info = await get_storage_info_async(printer.ip_address, printer.access_code)
  484. return storage_info or {"used_bytes": None, "free_bytes": None}
  485. # ============================================
  486. # MQTT Debug Logging Endpoints
  487. # ============================================
  488. @router.post("/{printer_id}/logging/enable")
  489. async def enable_mqtt_logging(printer_id: int, db: AsyncSession = Depends(get_db)):
  490. """Enable MQTT message logging for a printer."""
  491. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  492. printer = result.scalar_one_or_none()
  493. if not printer:
  494. raise HTTPException(404, "Printer not found")
  495. success = printer_manager.enable_logging(printer_id, True)
  496. if not success:
  497. raise HTTPException(400, "Printer not connected")
  498. return {"logging_enabled": True}
  499. @router.post("/{printer_id}/logging/disable")
  500. async def disable_mqtt_logging(printer_id: int, db: AsyncSession = Depends(get_db)):
  501. """Disable MQTT message logging for a printer."""
  502. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  503. printer = result.scalar_one_or_none()
  504. if not printer:
  505. raise HTTPException(404, "Printer not found")
  506. success = printer_manager.enable_logging(printer_id, False)
  507. if not success:
  508. raise HTTPException(400, "Printer not connected")
  509. return {"logging_enabled": False}
  510. @router.get("/{printer_id}/logging")
  511. async def get_mqtt_logs(printer_id: int, db: AsyncSession = Depends(get_db)):
  512. """Get MQTT message logs for a printer."""
  513. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  514. printer = result.scalar_one_or_none()
  515. if not printer:
  516. raise HTTPException(404, "Printer not found")
  517. logs = printer_manager.get_logs(printer_id)
  518. return {
  519. "logging_enabled": printer_manager.is_logging_enabled(printer_id),
  520. "logs": [
  521. {
  522. "timestamp": log.timestamp,
  523. "topic": log.topic,
  524. "direction": log.direction,
  525. "payload": log.payload,
  526. }
  527. for log in logs
  528. ],
  529. }
  530. @router.delete("/{printer_id}/logging")
  531. async def clear_mqtt_logs(printer_id: int, db: AsyncSession = Depends(get_db)):
  532. """Clear MQTT message logs for a printer."""
  533. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  534. printer = result.scalar_one_or_none()
  535. if not printer:
  536. raise HTTPException(404, "Printer not found")
  537. printer_manager.clear_logs(printer_id)
  538. return {"status": "cleared"}
  539. # ============================================
  540. # Print Options (AI Detection) Endpoints
  541. # ============================================
  542. @router.post("/{printer_id}/print-options")
  543. async def set_print_option(
  544. printer_id: int,
  545. module_name: str,
  546. enabled: bool,
  547. print_halt: bool = True,
  548. sensitivity: str = "medium",
  549. db: AsyncSession = Depends(get_db),
  550. ):
  551. """Set an AI detection / print option on the printer.
  552. Valid module_name values:
  553. - spaghetti_detector: Spaghetti detection
  554. - first_layer_inspector: First layer inspection
  555. - printing_monitor: AI print quality monitoring
  556. - buildplate_marker_detector: Build plate marker detection
  557. - allow_skip_parts: Allow skipping failed parts
  558. """
  559. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  560. printer = result.scalar_one_or_none()
  561. if not printer:
  562. raise HTTPException(404, "Printer not found")
  563. client = printer_manager.get_client(printer_id)
  564. if not client or not client.state.connected:
  565. raise HTTPException(400, "Printer not connected")
  566. # Validate module_name
  567. valid_modules = [
  568. "spaghetti_detector",
  569. "first_layer_inspector",
  570. "printing_monitor",
  571. "buildplate_marker_detector",
  572. "allow_skip_parts",
  573. "pileup_detector",
  574. "clump_detector",
  575. "airprint_detector",
  576. "auto_recovery_step_loss",
  577. ]
  578. if module_name not in valid_modules:
  579. raise HTTPException(400, f"Invalid module_name. Must be one of: {valid_modules}")
  580. # Validate sensitivity
  581. valid_sensitivities = ["low", "medium", "high", "never_halt"]
  582. if sensitivity not in valid_sensitivities:
  583. raise HTTPException(400, f"Invalid sensitivity. Must be one of: {valid_sensitivities}")
  584. success = client.set_xcam_option(
  585. module_name=module_name,
  586. enabled=enabled,
  587. print_halt=print_halt,
  588. sensitivity=sensitivity,
  589. )
  590. if not success:
  591. raise HTTPException(500, "Failed to send command to printer")
  592. return {
  593. "success": True,
  594. "module_name": module_name,
  595. "enabled": enabled,
  596. "print_halt": print_halt,
  597. "sensitivity": sensitivity,
  598. }
  599. # ============================================
  600. # Calibration
  601. # ============================================
  602. @router.post("/{printer_id}/calibration")
  603. async def start_calibration(
  604. printer_id: int,
  605. bed_leveling: bool = False,
  606. vibration: bool = False,
  607. motor_noise: bool = False,
  608. nozzle_offset: bool = False,
  609. high_temp_heatbed: bool = False,
  610. db: AsyncSession = Depends(get_db),
  611. ):
  612. """Start printer calibration with selected options.
  613. At least one option must be selected.
  614. Options:
  615. - bed_leveling: Run bed leveling calibration
  616. - vibration: Run vibration compensation calibration
  617. - motor_noise: Run motor noise cancellation calibration
  618. - nozzle_offset: Run nozzle offset calibration (dual nozzle printers)
  619. - high_temp_heatbed: Run high-temperature heatbed calibration
  620. """
  621. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  622. printer = result.scalar_one_or_none()
  623. if not printer:
  624. raise HTTPException(404, "Printer not found")
  625. client = printer_manager.get_client(printer_id)
  626. if not client or not client.state.connected:
  627. raise HTTPException(400, "Printer not connected")
  628. # Check that at least one option is selected
  629. if not any([bed_leveling, vibration, motor_noise, nozzle_offset, high_temp_heatbed]):
  630. raise HTTPException(400, "At least one calibration option must be selected")
  631. success = client.start_calibration(
  632. bed_leveling=bed_leveling,
  633. vibration=vibration,
  634. motor_noise=motor_noise,
  635. nozzle_offset=nozzle_offset,
  636. high_temp_heatbed=high_temp_heatbed,
  637. )
  638. if not success:
  639. raise HTTPException(500, "Failed to send calibration command to printer")
  640. return {
  641. "success": True,
  642. "bed_leveling": bed_leveling,
  643. "vibration": vibration,
  644. "motor_noise": motor_noise,
  645. "nozzle_offset": nozzle_offset,
  646. "high_temp_heatbed": high_temp_heatbed,
  647. }
  648. # ============================================================================
  649. # Slot Preset Mapping Endpoints
  650. # ============================================================================
  651. @router.get("/{printer_id}/slot-presets")
  652. async def get_slot_presets(
  653. printer_id: int,
  654. db: AsyncSession = Depends(get_db),
  655. ):
  656. """Get all saved slot-to-preset mappings for a printer."""
  657. result = await db.execute(
  658. select(SlotPresetMapping).where(SlotPresetMapping.printer_id == printer_id)
  659. )
  660. mappings = result.scalars().all()
  661. return {
  662. mapping.ams_id * 4 + mapping.tray_id: {
  663. "ams_id": mapping.ams_id,
  664. "tray_id": mapping.tray_id,
  665. "preset_id": mapping.preset_id,
  666. "preset_name": mapping.preset_name,
  667. }
  668. for mapping in mappings
  669. }
  670. @router.get("/{printer_id}/slot-presets/{ams_id}/{tray_id}")
  671. async def get_slot_preset(
  672. printer_id: int,
  673. ams_id: int,
  674. tray_id: int,
  675. db: AsyncSession = Depends(get_db),
  676. ):
  677. """Get the saved preset for a specific slot."""
  678. result = await db.execute(
  679. select(SlotPresetMapping).where(
  680. SlotPresetMapping.printer_id == printer_id,
  681. SlotPresetMapping.ams_id == ams_id,
  682. SlotPresetMapping.tray_id == tray_id,
  683. )
  684. )
  685. mapping = result.scalar_one_or_none()
  686. if not mapping:
  687. return None
  688. return {
  689. "ams_id": mapping.ams_id,
  690. "tray_id": mapping.tray_id,
  691. "preset_id": mapping.preset_id,
  692. "preset_name": mapping.preset_name,
  693. }
  694. @router.put("/{printer_id}/slot-presets/{ams_id}/{tray_id}")
  695. async def save_slot_preset(
  696. printer_id: int,
  697. ams_id: int,
  698. tray_id: int,
  699. preset_id: str,
  700. preset_name: str,
  701. db: AsyncSession = Depends(get_db),
  702. ):
  703. """Save a preset mapping for a specific slot."""
  704. # Check printer exists
  705. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  706. if not result.scalar_one_or_none():
  707. raise HTTPException(404, "Printer not found")
  708. # Check for existing mapping
  709. result = await db.execute(
  710. select(SlotPresetMapping).where(
  711. SlotPresetMapping.printer_id == printer_id,
  712. SlotPresetMapping.ams_id == ams_id,
  713. SlotPresetMapping.tray_id == tray_id,
  714. )
  715. )
  716. mapping = result.scalar_one_or_none()
  717. if mapping:
  718. # Update existing
  719. mapping.preset_id = preset_id
  720. mapping.preset_name = preset_name
  721. else:
  722. # Create new
  723. mapping = SlotPresetMapping(
  724. printer_id=printer_id,
  725. ams_id=ams_id,
  726. tray_id=tray_id,
  727. preset_id=preset_id,
  728. preset_name=preset_name,
  729. )
  730. db.add(mapping)
  731. await db.commit()
  732. await db.refresh(mapping)
  733. return {
  734. "ams_id": mapping.ams_id,
  735. "tray_id": mapping.tray_id,
  736. "preset_id": mapping.preset_id,
  737. "preset_name": mapping.preset_name,
  738. }
  739. @router.delete("/{printer_id}/slot-presets/{ams_id}/{tray_id}")
  740. async def delete_slot_preset(
  741. printer_id: int,
  742. ams_id: int,
  743. tray_id: int,
  744. db: AsyncSession = Depends(get_db),
  745. ):
  746. """Delete a saved preset mapping for a slot."""
  747. result = await db.execute(
  748. select(SlotPresetMapping).where(
  749. SlotPresetMapping.printer_id == printer_id,
  750. SlotPresetMapping.ams_id == ams_id,
  751. SlotPresetMapping.tray_id == tray_id,
  752. )
  753. )
  754. mapping = result.scalar_one_or_none()
  755. if mapping:
  756. await db.delete(mapping)
  757. await db.commit()
  758. return {"success": True}