kprofiles.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400
  1. """API routes for K-profile (pressure advance) management."""
  2. import asyncio
  3. import logging
  4. from fastapi import APIRouter, Depends, HTTPException
  5. from sqlalchemy import select
  6. from sqlalchemy.ext.asyncio import AsyncSession
  7. from backend.app.core.auth import RequirePermissionIfAuthEnabled
  8. from backend.app.core.database import get_db
  9. from backend.app.core.permissions import Permission
  10. from backend.app.models.kprofile_note import KProfileNote as KProfileNoteModel
  11. from backend.app.models.printer import Printer
  12. from backend.app.models.user import User
  13. from backend.app.schemas.kprofile import (
  14. KProfile,
  15. KProfileCreate,
  16. KProfileDelete,
  17. KProfileNote,
  18. KProfileNoteResponse,
  19. KProfilesResponse,
  20. )
  21. from backend.app.services.printer_manager import printer_manager
  22. logger = logging.getLogger(__name__)
  23. router = APIRouter(prefix="/printers/{printer_id}/kprofiles", tags=["kprofiles"])
  24. @router.get("/", response_model=KProfilesResponse)
  25. async def get_kprofiles(
  26. printer_id: int,
  27. nozzle_diameter: str = "0.4",
  28. db: AsyncSession = Depends(get_db),
  29. _: User | None = RequirePermissionIfAuthEnabled(Permission.KPROFILES_READ),
  30. ):
  31. """Get K-profiles from a printer.
  32. Args:
  33. printer_id: ID of the printer
  34. nozzle_diameter: Filter by nozzle diameter (default: "0.4")
  35. """
  36. # Check printer exists
  37. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  38. printer = result.scalar_one_or_none()
  39. if not printer:
  40. raise HTTPException(404, "Printer not found")
  41. # Get MQTT client for printer
  42. client = printer_manager.get_client(printer_id)
  43. if not client or not client.state.connected:
  44. raise HTTPException(400, "Printer not connected")
  45. # Request K-profiles from printer
  46. profiles = await client.get_kprofiles(nozzle_diameter=nozzle_diameter)
  47. # Convert from MQTT dataclass to Pydantic schema
  48. return KProfilesResponse(
  49. profiles=[
  50. KProfile(
  51. slot_id=p.slot_id,
  52. extruder_id=p.extruder_id,
  53. nozzle_id=p.nozzle_id,
  54. nozzle_diameter=p.nozzle_diameter,
  55. filament_id=p.filament_id,
  56. name=p.name,
  57. k_value=p.k_value,
  58. n_coef=p.n_coef,
  59. ams_id=p.ams_id,
  60. tray_id=p.tray_id,
  61. setting_id=p.setting_id,
  62. )
  63. for p in profiles
  64. ],
  65. nozzle_diameter=nozzle_diameter,
  66. )
  67. @router.post("/", response_model=dict)
  68. async def set_kprofile(
  69. printer_id: int,
  70. profile: KProfileCreate,
  71. db: AsyncSession = Depends(get_db),
  72. _: User | None = RequirePermissionIfAuthEnabled(Permission.KPROFILES_UPDATE),
  73. ):
  74. """Create or update a K-profile on the printer.
  75. For H2D edits (slot_id > 0), this performs an in-place edit using cali_idx.
  76. For other printers or new profiles, this adds a new profile.
  77. Args:
  78. printer_id: ID of the printer
  79. profile: K-profile data to set
  80. """
  81. is_edit = profile.slot_id > 0
  82. operation = "edit" if is_edit else "add"
  83. logger.info(
  84. f"[API] set_kprofile ({operation}): printer={printer_id}, slot_id={profile.slot_id}, "
  85. f"extruder_id={profile.extruder_id}, nozzle_id={profile.nozzle_id}, "
  86. f"name={profile.name}, filament_id={profile.filament_id}, k_value={profile.k_value}"
  87. )
  88. # Check printer exists
  89. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  90. printer = result.scalar_one_or_none()
  91. if not printer:
  92. raise HTTPException(404, "Printer not found")
  93. # Get MQTT client for printer
  94. client = printer_manager.get_client(printer_id)
  95. if not client or not client.state.connected:
  96. raise HTTPException(400, "Printer not connected")
  97. # Detect dual-nozzle for the in-place edit format. Runtime detection from
  98. # device.extruder.info beats serial-prefix heuristics — H2S shares prefix
  99. # "094" with H2D but is single-nozzle (#1386). Model name is the fallback
  100. # for the brief window after connect before push data arrives.
  101. is_dual_nozzle = client._is_dual_nozzle or (
  102. printer.model and printer.model.upper().strip() in ("H2D", "H2D PRO", "H2DPRO", "H2C", "X2D")
  103. )
  104. if is_edit and is_dual_nozzle:
  105. # Dual-nozzle in-place edit: use cali_idx with slot_id=0 and empty setting_id
  106. logger.info("[API] Dual-nozzle in-place edit: cali_idx=%s", profile.slot_id)
  107. success = client.set_kprofile(
  108. filament_id=profile.filament_id,
  109. name=profile.name,
  110. k_value=profile.k_value,
  111. nozzle_diameter=profile.nozzle_diameter,
  112. nozzle_id=profile.nozzle_id,
  113. extruder_id=profile.extruder_id,
  114. setting_id=None,
  115. slot_id=0,
  116. cali_idx=profile.slot_id, # Pass the original slot for in-place edit
  117. )
  118. elif is_edit:
  119. # Single-nozzle edit: use delete + add approach
  120. logger.info("[API] Edit: deleting existing profile slot_id=%s", profile.slot_id)
  121. delete_success = client.delete_kprofile(
  122. cali_idx=profile.slot_id,
  123. filament_id=profile.filament_id,
  124. nozzle_id=profile.nozzle_id,
  125. nozzle_diameter=profile.nozzle_diameter,
  126. extruder_id=profile.extruder_id,
  127. setting_id=profile.setting_id,
  128. )
  129. if not delete_success:
  130. raise HTTPException(500, "Failed to delete existing K-profile for edit")
  131. # Wait for printer to process the delete before adding
  132. await asyncio.sleep(0.5)
  133. logger.info("[API] Edit: delete complete, now adding updated profile")
  134. success = client.set_kprofile(
  135. filament_id=profile.filament_id,
  136. name=profile.name,
  137. k_value=profile.k_value,
  138. nozzle_diameter=profile.nozzle_diameter,
  139. nozzle_id=profile.nozzle_id,
  140. extruder_id=profile.extruder_id,
  141. setting_id=None, # Generate new setting_id for add
  142. slot_id=0, # Always 0 for add (new profile)
  143. )
  144. else:
  145. # New profile: add with slot_id=0
  146. success = client.set_kprofile(
  147. filament_id=profile.filament_id,
  148. name=profile.name,
  149. k_value=profile.k_value,
  150. nozzle_diameter=profile.nozzle_diameter,
  151. nozzle_id=profile.nozzle_id,
  152. extruder_id=profile.extruder_id,
  153. setting_id=None, # Generate new setting_id for add
  154. slot_id=0, # Always 0 for add (new profile)
  155. )
  156. if not success:
  157. raise HTTPException(500, "Failed to send K-profile command")
  158. message = "K-profile updated successfully" if is_edit else "K-profile added successfully"
  159. return {"success": True, "message": message}
  160. @router.post("/batch", response_model=dict)
  161. async def set_kprofiles_batch(
  162. printer_id: int,
  163. profiles: list[KProfileCreate],
  164. db: AsyncSession = Depends(get_db),
  165. _: User | None = RequirePermissionIfAuthEnabled(Permission.KPROFILES_UPDATE),
  166. ):
  167. """Create multiple K-profiles in a single command (for dual-nozzle).
  168. This sends all profiles in one MQTT command, which is more reliable
  169. for dual-nozzle printers that may not handle sequential commands well.
  170. Args:
  171. printer_id: ID of the printer
  172. profiles: List of K-profiles to set
  173. """
  174. if not profiles:
  175. raise HTTPException(400, "No profiles provided")
  176. logger.info("[API] set_kprofiles_batch: printer=%s, %s profiles", printer_id, len(profiles))
  177. for p in profiles:
  178. logger.info(" - extruder_id=%s, name=%s, k_value=%s", p.extruder_id, p.name, p.k_value)
  179. # Check printer exists
  180. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  181. printer = result.scalar_one_or_none()
  182. if not printer:
  183. raise HTTPException(404, "Printer not found")
  184. # Get MQTT client for printer
  185. client = printer_manager.get_client(printer_id)
  186. if not client or not client.state.connected:
  187. raise HTTPException(400, "Printer not connected")
  188. # Build list of profile dicts for batch command
  189. profile_dicts = [
  190. {
  191. "filament_id": p.filament_id,
  192. "name": p.name,
  193. "k_value": p.k_value,
  194. "nozzle_id": p.nozzle_id,
  195. "extruder_id": p.extruder_id,
  196. "setting_id": p.setting_id,
  197. "slot_id": p.slot_id,
  198. }
  199. for p in profiles
  200. ]
  201. # Get nozzle_diameter from first profile (all should have same)
  202. nozzle_diameter = profiles[0].nozzle_diameter
  203. success = client.set_kprofiles_batch(profile_dicts, nozzle_diameter)
  204. if not success:
  205. raise HTTPException(500, "Failed to send K-profiles batch command")
  206. return {"success": True, "message": f"Added {len(profiles)} K-profiles"}
  207. @router.delete("/", response_model=dict)
  208. async def delete_kprofile(
  209. printer_id: int,
  210. profile: KProfileDelete,
  211. db: AsyncSession = Depends(get_db),
  212. _: User | None = RequirePermissionIfAuthEnabled(Permission.KPROFILES_DELETE),
  213. ):
  214. """Delete a K-profile from the printer.
  215. Args:
  216. printer_id: ID of the printer
  217. profile: K-profile identification data for deletion
  218. """
  219. # Check printer exists
  220. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  221. printer = result.scalar_one_or_none()
  222. if not printer:
  223. raise HTTPException(404, "Printer not found")
  224. # Get MQTT client for printer
  225. client = printer_manager.get_client(printer_id)
  226. if not client or not client.state.connected:
  227. raise HTTPException(400, "Printer not connected")
  228. # Send the delete command to printer
  229. logger.info(
  230. f"[API] delete_kprofile: printer={printer_id}, slot_id={profile.slot_id}, "
  231. f"setting_id={profile.setting_id}, filament_id={profile.filament_id}"
  232. )
  233. success = client.delete_kprofile(
  234. cali_idx=profile.slot_id,
  235. filament_id=profile.filament_id,
  236. nozzle_id=profile.nozzle_id,
  237. nozzle_diameter=profile.nozzle_diameter,
  238. extruder_id=profile.extruder_id,
  239. setting_id=profile.setting_id,
  240. )
  241. if not success:
  242. raise HTTPException(500, "Failed to send K-profile delete command")
  243. # Wait for printer to process the delete before frontend refetches
  244. await asyncio.sleep(0.5)
  245. return {"success": True, "message": "K-profile deleted successfully"}
  246. @router.get("/notes", response_model=KProfileNoteResponse)
  247. async def get_kprofile_notes(
  248. printer_id: int,
  249. db: AsyncSession = Depends(get_db),
  250. _: User | None = RequirePermissionIfAuthEnabled(Permission.KPROFILES_READ),
  251. ):
  252. """Get all K-profile notes for a printer.
  253. Notes are stored locally since printers don't support notes.
  254. Args:
  255. printer_id: ID of the printer
  256. """
  257. # Check printer exists
  258. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  259. printer = result.scalar_one_or_none()
  260. if not printer:
  261. raise HTTPException(404, "Printer not found")
  262. # Get all notes for this printer
  263. result = await db.execute(select(KProfileNoteModel).where(KProfileNoteModel.printer_id == printer_id))
  264. notes = result.scalars().all()
  265. # Return as a dictionary mapping setting_id -> note
  266. return KProfileNoteResponse(notes={note.setting_id: note.note for note in notes})
  267. @router.put("/notes", response_model=dict)
  268. async def set_kprofile_note(
  269. printer_id: int,
  270. note_data: KProfileNote,
  271. db: AsyncSession = Depends(get_db),
  272. _: User | None = RequirePermissionIfAuthEnabled(Permission.KPROFILES_UPDATE),
  273. ):
  274. """Set or update a note for a K-profile.
  275. Args:
  276. printer_id: ID of the printer
  277. note_data: The note data (setting_id and note content)
  278. """
  279. # Check printer exists
  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. # Find existing note or create new one
  285. result = await db.execute(
  286. select(KProfileNoteModel).where(
  287. KProfileNoteModel.printer_id == printer_id,
  288. KProfileNoteModel.setting_id == note_data.setting_id,
  289. )
  290. )
  291. existing_note = result.scalar_one_or_none()
  292. if note_data.note.strip():
  293. # Save or update note
  294. if existing_note:
  295. existing_note.note = note_data.note
  296. else:
  297. new_note = KProfileNoteModel(
  298. printer_id=printer_id,
  299. setting_id=note_data.setting_id,
  300. note=note_data.note,
  301. )
  302. db.add(new_note)
  303. await db.commit()
  304. return {"success": True, "message": "Note saved"}
  305. else:
  306. # Delete note if empty
  307. if existing_note:
  308. await db.delete(existing_note)
  309. await db.commit()
  310. return {"success": True, "message": "Note deleted"}
  311. @router.delete("/notes/{setting_id}", response_model=dict)
  312. async def delete_kprofile_note(
  313. printer_id: int,
  314. setting_id: str,
  315. db: AsyncSession = Depends(get_db),
  316. _: User | None = RequirePermissionIfAuthEnabled(Permission.KPROFILES_DELETE),
  317. ):
  318. """Delete a note for a K-profile.
  319. Args:
  320. printer_id: ID of the printer
  321. setting_id: The setting_id of the K-profile
  322. """
  323. # Check printer exists
  324. result = await db.execute(select(Printer).where(Printer.id == printer_id))
  325. printer = result.scalar_one_or_none()
  326. if not printer:
  327. raise HTTPException(404, "Printer not found")
  328. # Find and delete the note
  329. result = await db.execute(
  330. select(KProfileNoteModel).where(
  331. KProfileNoteModel.printer_id == printer_id,
  332. KProfileNoteModel.setting_id == setting_id,
  333. )
  334. )
  335. existing_note = result.scalar_one_or_none()
  336. if existing_note:
  337. await db.delete(existing_note)
  338. await db.commit()
  339. return {"success": True, "message": "Note deleted"}