printers.py 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898
  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. )
  277. @router.post("/{printer_id}/connect")
  278. async def connect_printer(printer_id: int, db: AsyncSession = Depends(get_db)):
  279. """Manually connect to a printer."""
  280. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  281. printer = result.scalar_one_or_none()
  282. if not printer:
  283. raise HTTPException(404, "Printer not found")
  284. success = await printer_manager.connect_printer(printer)
  285. return {"connected": success}
  286. @router.post("/{printer_id}/disconnect")
  287. async def disconnect_printer(printer_id: int, db: AsyncSession = Depends(get_db)):
  288. """Manually disconnect from a printer."""
  289. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  290. printer = result.scalar_one_or_none()
  291. if not printer:
  292. raise HTTPException(404, "Printer not found")
  293. printer_manager.disconnect_printer(printer_id)
  294. return {"connected": False}
  295. @router.post("/test")
  296. async def test_printer_connection(
  297. ip_address: str,
  298. serial_number: str,
  299. access_code: str,
  300. ):
  301. """Test connection to a printer without saving."""
  302. result = await printer_manager.test_connection(
  303. ip_address=ip_address,
  304. serial_number=serial_number,
  305. access_code=access_code,
  306. )
  307. return result
  308. # Cache for cover images (printer_id -> (gcode_file, image_bytes))
  309. _cover_cache: dict[int, tuple[str, bytes]] = {}
  310. @router.get("/{printer_id}/cover")
  311. async def get_printer_cover(printer_id: int, db: AsyncSession = Depends(get_db)):
  312. """Get the cover image for the current print job."""
  313. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  314. printer = result.scalar_one_or_none()
  315. if not printer:
  316. raise HTTPException(404, "Printer not found")
  317. state = printer_manager.get_status(printer_id)
  318. if not state:
  319. raise HTTPException(404, "Printer not connected")
  320. # Use subtask_name as the 3MF filename (gcode_file is the path inside the 3MF)
  321. subtask_name = state.subtask_name
  322. if not subtask_name:
  323. raise HTTPException(404, f"No subtask_name in printer state (state={state.state})")
  324. # Check cache
  325. if printer_id in _cover_cache:
  326. cached_file, cached_image = _cover_cache[printer_id]
  327. if cached_file == subtask_name:
  328. return Response(content=cached_image, media_type="image/png")
  329. # Build 3MF filename from subtask_name
  330. # Bambu printers store files as "name.gcode.3mf"
  331. filename = subtask_name
  332. if not filename.endswith(".3mf"):
  333. filename = filename + ".gcode.3mf"
  334. # Try to download the 3MF file from printer
  335. temp_path = settings.archive_dir / "temp" / f"cover_{printer_id}_{filename}"
  336. temp_path.parent.mkdir(parents=True, exist_ok=True)
  337. remote_paths = [
  338. f"/{filename}", # Root directory (most common)
  339. f"/cache/{filename}",
  340. f"/model/{filename}",
  341. f"/data/{filename}",
  342. ]
  343. logger.info(f"Trying to download cover for '{filename}' from {printer.ip_address}")
  344. try:
  345. downloaded = await download_file_try_paths_async(
  346. printer.ip_address,
  347. printer.access_code,
  348. remote_paths,
  349. temp_path,
  350. )
  351. except Exception as e:
  352. logger.error(f"FTP download exception: {e}")
  353. raise HTTPException(500, f"FTP download failed: {e}")
  354. if not downloaded:
  355. raise HTTPException(404, f"Could not download 3MF file '{filename}' from printer {printer.ip_address}. Tried: {remote_paths}")
  356. # Verify file actually exists and has content
  357. if not temp_path.exists():
  358. raise HTTPException(500, f"Download reported success but file not found: {temp_path}")
  359. file_size = temp_path.stat().st_size
  360. logger.info(f"Downloaded file size: {file_size} bytes")
  361. if file_size == 0:
  362. temp_path.unlink()
  363. raise HTTPException(500, f"Downloaded file is empty: {filename}")
  364. try:
  365. # Extract thumbnail from 3MF (which is a ZIP file)
  366. try:
  367. zf = zipfile.ZipFile(temp_path, 'r')
  368. except zipfile.BadZipFile as e:
  369. raise HTTPException(500, f"Downloaded file is not a valid 3MF/ZIP: {e}")
  370. except Exception as e:
  371. raise HTTPException(500, f"Failed to open 3MF file: {e}")
  372. try:
  373. # Try common thumbnail paths in 3MF files
  374. thumbnail_paths = [
  375. "Metadata/plate_1.png",
  376. "Metadata/thumbnail.png",
  377. "Metadata/plate_1_small.png",
  378. "Thumbnails/thumbnail.png",
  379. "thumbnail.png",
  380. ]
  381. for thumb_path in thumbnail_paths:
  382. try:
  383. image_data = zf.read(thumb_path)
  384. # Cache the result
  385. _cover_cache[printer_id] = (subtask_name, image_data)
  386. return Response(content=image_data, media_type="image/png")
  387. except KeyError:
  388. continue
  389. # If no specific thumbnail found, try any PNG in Metadata
  390. for name in zf.namelist():
  391. if name.startswith("Metadata/") and name.endswith(".png"):
  392. image_data = zf.read(name)
  393. _cover_cache[printer_id] = (subtask_name, image_data)
  394. return Response(content=image_data, media_type="image/png")
  395. raise HTTPException(404, "No thumbnail found in 3MF file")
  396. finally:
  397. zf.close()
  398. finally:
  399. if temp_path.exists():
  400. temp_path.unlink()
  401. # ============================================
  402. # File Manager Endpoints
  403. # ============================================
  404. @router.get("/{printer_id}/files")
  405. async def list_printer_files(
  406. printer_id: int,
  407. path: str = "/",
  408. db: AsyncSession = Depends(get_db),
  409. ):
  410. """List files on the printer at the specified path."""
  411. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  412. printer = result.scalar_one_or_none()
  413. if not printer:
  414. raise HTTPException(404, "Printer not found")
  415. files = await list_files_async(printer.ip_address, printer.access_code, path)
  416. # Add full path to each file
  417. for f in files:
  418. f["path"] = f"{path.rstrip('/')}/{f['name']}" if path != "/" else f"/{f['name']}"
  419. return {
  420. "path": path,
  421. "files": files,
  422. }
  423. @router.get("/{printer_id}/files/download")
  424. async def download_printer_file(
  425. printer_id: int,
  426. path: str,
  427. db: AsyncSession = Depends(get_db),
  428. ):
  429. """Download a file from the printer."""
  430. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  431. printer = result.scalar_one_or_none()
  432. if not printer:
  433. raise HTTPException(404, "Printer not found")
  434. data = await download_file_bytes_async(printer.ip_address, printer.access_code, path)
  435. if data is None:
  436. raise HTTPException(404, f"File not found: {path}")
  437. # Determine content type based on extension
  438. filename = path.split("/")[-1]
  439. ext = filename.lower().split(".")[-1] if "." in filename else ""
  440. content_types = {
  441. "3mf": "application/vnd.ms-package.3dmanufacturing-3dmodel+xml",
  442. "gcode": "text/plain",
  443. "mp4": "video/mp4",
  444. "avi": "video/x-msvideo",
  445. "png": "image/png",
  446. "jpg": "image/jpeg",
  447. "jpeg": "image/jpeg",
  448. "json": "application/json",
  449. "txt": "text/plain",
  450. }
  451. content_type = content_types.get(ext, "application/octet-stream")
  452. return Response(
  453. content=data,
  454. media_type=content_type,
  455. headers={"Content-Disposition": f'attachment; filename="{filename}"'},
  456. )
  457. @router.delete("/{printer_id}/files")
  458. async def delete_printer_file(
  459. printer_id: int,
  460. path: str,
  461. db: AsyncSession = Depends(get_db),
  462. ):
  463. """Delete a file from the printer."""
  464. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  465. printer = result.scalar_one_or_none()
  466. if not printer:
  467. raise HTTPException(404, "Printer not found")
  468. success = await delete_file_async(printer.ip_address, printer.access_code, path)
  469. if not success:
  470. raise HTTPException(500, f"Failed to delete file: {path}")
  471. return {"status": "deleted", "path": path}
  472. @router.get("/{printer_id}/storage")
  473. async def get_printer_storage(
  474. printer_id: int,
  475. db: AsyncSession = Depends(get_db),
  476. ):
  477. """Get storage information from the printer."""
  478. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  479. printer = result.scalar_one_or_none()
  480. if not printer:
  481. raise HTTPException(404, "Printer not found")
  482. storage_info = await get_storage_info_async(printer.ip_address, printer.access_code)
  483. return storage_info or {"used_bytes": None, "free_bytes": None}
  484. # ============================================
  485. # MQTT Debug Logging Endpoints
  486. # ============================================
  487. @router.post("/{printer_id}/logging/enable")
  488. async def enable_mqtt_logging(printer_id: int, db: AsyncSession = Depends(get_db)):
  489. """Enable MQTT message logging for a printer."""
  490. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  491. printer = result.scalar_one_or_none()
  492. if not printer:
  493. raise HTTPException(404, "Printer not found")
  494. success = printer_manager.enable_logging(printer_id, True)
  495. if not success:
  496. raise HTTPException(400, "Printer not connected")
  497. return {"logging_enabled": True}
  498. @router.post("/{printer_id}/logging/disable")
  499. async def disable_mqtt_logging(printer_id: int, db: AsyncSession = Depends(get_db)):
  500. """Disable MQTT message logging for a printer."""
  501. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  502. printer = result.scalar_one_or_none()
  503. if not printer:
  504. raise HTTPException(404, "Printer not found")
  505. success = printer_manager.enable_logging(printer_id, False)
  506. if not success:
  507. raise HTTPException(400, "Printer not connected")
  508. return {"logging_enabled": False}
  509. @router.get("/{printer_id}/logging")
  510. async def get_mqtt_logs(printer_id: int, db: AsyncSession = Depends(get_db)):
  511. """Get MQTT message logs for a printer."""
  512. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  513. printer = result.scalar_one_or_none()
  514. if not printer:
  515. raise HTTPException(404, "Printer not found")
  516. logs = printer_manager.get_logs(printer_id)
  517. return {
  518. "logging_enabled": printer_manager.is_logging_enabled(printer_id),
  519. "logs": [
  520. {
  521. "timestamp": log.timestamp,
  522. "topic": log.topic,
  523. "direction": log.direction,
  524. "payload": log.payload,
  525. }
  526. for log in logs
  527. ],
  528. }
  529. @router.delete("/{printer_id}/logging")
  530. async def clear_mqtt_logs(printer_id: int, db: AsyncSession = Depends(get_db)):
  531. """Clear MQTT message logs for a printer."""
  532. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  533. printer = result.scalar_one_or_none()
  534. if not printer:
  535. raise HTTPException(404, "Printer not found")
  536. printer_manager.clear_logs(printer_id)
  537. return {"status": "cleared"}
  538. # ============================================
  539. # Print Options (AI Detection) Endpoints
  540. # ============================================
  541. @router.post("/{printer_id}/print-options")
  542. async def set_print_option(
  543. printer_id: int,
  544. module_name: str,
  545. enabled: bool,
  546. print_halt: bool = True,
  547. sensitivity: str = "medium",
  548. db: AsyncSession = Depends(get_db),
  549. ):
  550. """Set an AI detection / print option on the printer.
  551. Valid module_name values:
  552. - spaghetti_detector: Spaghetti detection
  553. - first_layer_inspector: First layer inspection
  554. - printing_monitor: AI print quality monitoring
  555. - buildplate_marker_detector: Build plate marker detection
  556. - allow_skip_parts: Allow skipping failed parts
  557. """
  558. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  559. printer = result.scalar_one_or_none()
  560. if not printer:
  561. raise HTTPException(404, "Printer not found")
  562. client = printer_manager.get_client(printer_id)
  563. if not client or not client.state.connected:
  564. raise HTTPException(400, "Printer not connected")
  565. # Validate module_name
  566. valid_modules = [
  567. "spaghetti_detector",
  568. "first_layer_inspector",
  569. "printing_monitor",
  570. "buildplate_marker_detector",
  571. "allow_skip_parts",
  572. "pileup_detector",
  573. "clump_detector",
  574. "airprint_detector",
  575. "auto_recovery_step_loss",
  576. ]
  577. if module_name not in valid_modules:
  578. raise HTTPException(400, f"Invalid module_name. Must be one of: {valid_modules}")
  579. # Validate sensitivity
  580. valid_sensitivities = ["low", "medium", "high", "never_halt"]
  581. if sensitivity not in valid_sensitivities:
  582. raise HTTPException(400, f"Invalid sensitivity. Must be one of: {valid_sensitivities}")
  583. success = client.set_xcam_option(
  584. module_name=module_name,
  585. enabled=enabled,
  586. print_halt=print_halt,
  587. sensitivity=sensitivity,
  588. )
  589. if not success:
  590. raise HTTPException(500, "Failed to send command to printer")
  591. return {
  592. "success": True,
  593. "module_name": module_name,
  594. "enabled": enabled,
  595. "print_halt": print_halt,
  596. "sensitivity": sensitivity,
  597. }
  598. # ============================================
  599. # Calibration
  600. # ============================================
  601. @router.post("/{printer_id}/calibration")
  602. async def start_calibration(
  603. printer_id: int,
  604. bed_leveling: bool = False,
  605. vibration: bool = False,
  606. motor_noise: bool = False,
  607. nozzle_offset: bool = False,
  608. high_temp_heatbed: bool = False,
  609. db: AsyncSession = Depends(get_db),
  610. ):
  611. """Start printer calibration with selected options.
  612. At least one option must be selected.
  613. Options:
  614. - bed_leveling: Run bed leveling calibration
  615. - vibration: Run vibration compensation calibration
  616. - motor_noise: Run motor noise cancellation calibration
  617. - nozzle_offset: Run nozzle offset calibration (dual nozzle printers)
  618. - high_temp_heatbed: Run high-temperature heatbed calibration
  619. """
  620. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  621. printer = result.scalar_one_or_none()
  622. if not printer:
  623. raise HTTPException(404, "Printer not found")
  624. client = printer_manager.get_client(printer_id)
  625. if not client or not client.state.connected:
  626. raise HTTPException(400, "Printer not connected")
  627. # Check that at least one option is selected
  628. if not any([bed_leveling, vibration, motor_noise, nozzle_offset, high_temp_heatbed]):
  629. raise HTTPException(400, "At least one calibration option must be selected")
  630. success = client.start_calibration(
  631. bed_leveling=bed_leveling,
  632. vibration=vibration,
  633. motor_noise=motor_noise,
  634. nozzle_offset=nozzle_offset,
  635. high_temp_heatbed=high_temp_heatbed,
  636. )
  637. if not success:
  638. raise HTTPException(500, "Failed to send calibration command to printer")
  639. return {
  640. "success": True,
  641. "bed_leveling": bed_leveling,
  642. "vibration": vibration,
  643. "motor_noise": motor_noise,
  644. "nozzle_offset": nozzle_offset,
  645. "high_temp_heatbed": high_temp_heatbed,
  646. }
  647. # ============================================================================
  648. # Slot Preset Mapping Endpoints
  649. # ============================================================================
  650. @router.get("/{printer_id}/slot-presets")
  651. async def get_slot_presets(
  652. printer_id: int,
  653. db: AsyncSession = Depends(get_db),
  654. ):
  655. """Get all saved slot-to-preset mappings for a printer."""
  656. result = await db.execute(
  657. select(SlotPresetMapping).where(SlotPresetMapping.printer_id == printer_id)
  658. )
  659. mappings = result.scalars().all()
  660. return {
  661. mapping.ams_id * 4 + mapping.tray_id: {
  662. "ams_id": mapping.ams_id,
  663. "tray_id": mapping.tray_id,
  664. "preset_id": mapping.preset_id,
  665. "preset_name": mapping.preset_name,
  666. }
  667. for mapping in mappings
  668. }
  669. @router.get("/{printer_id}/slot-presets/{ams_id}/{tray_id}")
  670. async def get_slot_preset(
  671. printer_id: int,
  672. ams_id: int,
  673. tray_id: int,
  674. db: AsyncSession = Depends(get_db),
  675. ):
  676. """Get the saved preset for a specific slot."""
  677. result = await db.execute(
  678. select(SlotPresetMapping).where(
  679. SlotPresetMapping.printer_id == printer_id,
  680. SlotPresetMapping.ams_id == ams_id,
  681. SlotPresetMapping.tray_id == tray_id,
  682. )
  683. )
  684. mapping = result.scalar_one_or_none()
  685. if not mapping:
  686. return None
  687. return {
  688. "ams_id": mapping.ams_id,
  689. "tray_id": mapping.tray_id,
  690. "preset_id": mapping.preset_id,
  691. "preset_name": mapping.preset_name,
  692. }
  693. @router.put("/{printer_id}/slot-presets/{ams_id}/{tray_id}")
  694. async def save_slot_preset(
  695. printer_id: int,
  696. ams_id: int,
  697. tray_id: int,
  698. preset_id: str,
  699. preset_name: str,
  700. db: AsyncSession = Depends(get_db),
  701. ):
  702. """Save a preset mapping for a specific slot."""
  703. # Check printer exists
  704. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  705. if not result.scalar_one_or_none():
  706. raise HTTPException(404, "Printer not found")
  707. # Check for existing mapping
  708. result = await db.execute(
  709. select(SlotPresetMapping).where(
  710. SlotPresetMapping.printer_id == printer_id,
  711. SlotPresetMapping.ams_id == ams_id,
  712. SlotPresetMapping.tray_id == tray_id,
  713. )
  714. )
  715. mapping = result.scalar_one_or_none()
  716. if mapping:
  717. # Update existing
  718. mapping.preset_id = preset_id
  719. mapping.preset_name = preset_name
  720. else:
  721. # Create new
  722. mapping = SlotPresetMapping(
  723. printer_id=printer_id,
  724. ams_id=ams_id,
  725. tray_id=tray_id,
  726. preset_id=preset_id,
  727. preset_name=preset_name,
  728. )
  729. db.add(mapping)
  730. await db.commit()
  731. await db.refresh(mapping)
  732. return {
  733. "ams_id": mapping.ams_id,
  734. "tray_id": mapping.tray_id,
  735. "preset_id": mapping.preset_id,
  736. "preset_name": mapping.preset_name,
  737. }
  738. @router.delete("/{printer_id}/slot-presets/{ams_id}/{tray_id}")
  739. async def delete_slot_preset(
  740. printer_id: int,
  741. ams_id: int,
  742. tray_id: int,
  743. db: AsyncSession = Depends(get_db),
  744. ):
  745. """Delete a saved preset mapping for a slot."""
  746. result = await db.execute(
  747. select(SlotPresetMapping).where(
  748. SlotPresetMapping.printer_id == printer_id,
  749. SlotPresetMapping.ams_id == ams_id,
  750. SlotPresetMapping.tray_id == tray_id,
  751. )
  752. )
  753. mapping = result.scalar_one_or_none()
  754. if mapping:
  755. await db.delete(mapping)
  756. await db.commit()
  757. return {"success": True}