printer_manager.py 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817
  1. import asyncio
  2. import logging
  3. import traceback
  4. from collections.abc import Callable
  5. from sqlalchemy import select
  6. from sqlalchemy.ext.asyncio import AsyncSession
  7. from backend.app.models.printer import Printer
  8. from backend.app.services.bambu_mqtt import BambuMQTTClient, MQTTLogEntry, PrinterState, get_stage_name
  9. logger = logging.getLogger(__name__)
  10. # Models that have a real chamber temperature sensor
  11. # Based on Home Assistant Bambu Lab integration
  12. # P1P/P1S and A1/A1Mini do NOT have chamber temp sensors
  13. # Includes both display names and internal codes from MQTT/SSDP
  14. CHAMBER_TEMP_SUPPORTED_MODELS = frozenset(
  15. [
  16. # Display names
  17. "X1",
  18. "X1C",
  19. "X1E", # X1 series
  20. "P2S", # P2 series
  21. "H2C",
  22. "H2D",
  23. "H2DPRO",
  24. "H2S", # H2 series
  25. # Internal codes (from MQTT/SSDP)
  26. "BL-P001", # X1/X1C
  27. "C13", # X1E
  28. "O1D", # H2D
  29. "O1C", # H2C
  30. "O1C2", # H2C (dual nozzle variant)
  31. "O1S", # H2S
  32. "O1E", # H2D Pro
  33. "O2D", # H2D Pro (alternate code)
  34. "N7", # P2S
  35. ]
  36. )
  37. # Models that may incorrectly report stg_cur=0 when idle (firmware bug)
  38. # Based on Home Assistant Bambu Lab integration observations
  39. # See: https://github.com/greghesp/ha-bambulab/blob/main/custom_components/bambu_lab/pybambu/models.py
  40. A1_MODELS = frozenset(
  41. [
  42. # Display names
  43. "A1",
  44. "A1 MINI",
  45. "A1-MINI",
  46. "A1MINI",
  47. # Internal codes (from MQTT/SSDP)
  48. "N1", # A1 Mini
  49. "N2S", # A1
  50. ]
  51. )
  52. # Models affected by the stg_cur=0 idle bug (firmware reports stg_cur=0 when idle,
  53. # which maps to "Printing" in STAGE_NAMES and overrides the correct IDLE state)
  54. STG_CUR_IDLE_BUG_MODELS = A1_MODELS | frozenset(
  55. [
  56. # Display names
  57. "P1P",
  58. "P1S",
  59. # Internal codes (from MQTT/SSDP)
  60. "C11", # P1P
  61. "C12", # P1S
  62. ]
  63. )
  64. def supports_chamber_temp(model: str | None) -> bool:
  65. """Check if a printer model has a real chamber temperature sensor.
  66. P1P, P1S, A1, and A1Mini do NOT have chamber temp sensors.
  67. The 'chamber_temper' value they report is meaningless.
  68. """
  69. if not model:
  70. return False
  71. # Normalize model name (uppercase, strip whitespace)
  72. model_upper = model.strip().upper()
  73. return model_upper in CHAMBER_TEMP_SUPPORTED_MODELS
  74. def has_stg_cur_idle_bug(model: str | None) -> bool:
  75. """Check if a printer model may incorrectly report stg_cur=0 when idle.
  76. Some firmware versions report stg_cur=0 (which maps to "Printing")
  77. even when the printer is idle. Originally observed on A1/A1 Mini via the
  78. Home Assistant Bambu Lab integration, also confirmed on P1S.
  79. """
  80. if not model:
  81. return False
  82. model_upper = model.strip().upper()
  83. return model_upper in STG_CUR_IDLE_BUG_MODELS
  84. # Minimum firmware versions for AMS drying support (confirmed via capture testing)
  85. # Keys are exact model names (upper-cased). Do NOT use substring matching — it would
  86. # incorrectly gate X1E (matched by "X1") and H2D Pro (matched by "H2D").
  87. _DRYING_MIN_FIRMWARE: dict[str, str] = {
  88. "H2D": "01.02.30.00",
  89. "X1": "01.09.00.00",
  90. "X1C": "01.09.00.00",
  91. "P1P": "01.08.00.00",
  92. "P1S": "01.08.00.00",
  93. }
  94. # Models that definitely don't support AMS drying (no AMS 2 Pro / AMS-HT compatibility)
  95. _DRYING_UNSUPPORTED_MODELS = frozenset(
  96. {"P2S", "A1", "A1MINI", "A1-MINI", "A1 MINI", "H2S", "H2C", "N7", "O1C", "O1C2", "O1S", "N1", "N2S"}
  97. )
  98. def supports_drying(model: str | None, firmware: str | None) -> bool:
  99. """Check if a printer model supports AMS drying commands.
  100. Known models with confirmed min firmware get version-gated.
  101. Known unsupported models are blocked.
  102. All other models (H2D Pro, X1E, future models) are allowed —
  103. the command fails gracefully with result: "fail" if unsupported.
  104. """
  105. if not model:
  106. return False
  107. model_upper = model.strip().upper()
  108. if model_upper in _DRYING_UNSUPPORTED_MODELS:
  109. return False
  110. if model_upper in _DRYING_MIN_FIRMWARE:
  111. return bool(firmware and firmware >= _DRYING_MIN_FIRMWARE[model_upper])
  112. # For all other models: allow
  113. return True
  114. class PrinterInfo:
  115. """Basic printer info for callbacks."""
  116. def __init__(self, name: str, serial_number: str):
  117. self.name = name
  118. self.serial_number = serial_number
  119. class PrinterManager:
  120. """Manager for multiple printer connections."""
  121. def __init__(self):
  122. self._clients: dict[int, BambuMQTTClient] = {}
  123. self._models: dict[int, str | None] = {} # Cache printer models for feature detection
  124. self._printer_info: dict[int, PrinterInfo] = {} # Cache printer name/serial for callbacks
  125. self._on_print_start: Callable[[int, dict], None] | None = None
  126. self._on_print_complete: Callable[[int, dict], None] | None = None
  127. self._on_status_change: Callable[[int, PrinterState], None] | None = None
  128. self._on_ams_change: Callable[[int, list], None] | None = None
  129. self._on_layer_change: Callable[[int, int], None] | None = None
  130. self._on_bed_temp_update: Callable[[int, float], None] | None = None
  131. self._loop: asyncio.AbstractEventLoop | None = None
  132. # Track who started the current print (Issue #206)
  133. self._current_print_user: dict[int, dict] = {} # {printer_id: {"user_id": int, "username": str}}
  134. # Track plate-cleared acknowledgments for queue flow
  135. self._plate_cleared: set[int] = set() # printer_ids where user confirmed plate is cleared
  136. def get_printer(self, printer_id: int) -> PrinterInfo | None:
  137. """Get printer info by ID."""
  138. return self._printer_info.get(printer_id)
  139. def set_current_print_user(self, printer_id: int, user_id: int, username: str):
  140. """Track who started the current print (Issue #206)."""
  141. self._current_print_user[printer_id] = {"user_id": user_id, "username": username}
  142. def get_current_print_user(self, printer_id: int) -> dict | None:
  143. """Get the user who started the current print (Issue #206)."""
  144. return self._current_print_user.get(printer_id)
  145. def clear_current_print_user(self, printer_id: int):
  146. """Clear the current print user when print completes (Issue #206)."""
  147. self._current_print_user.pop(printer_id, None)
  148. def set_plate_cleared(self, printer_id: int):
  149. """Mark that user has cleared the build plate for this printer."""
  150. self._plate_cleared.add(printer_id)
  151. def is_plate_cleared(self, printer_id: int) -> bool:
  152. """Check if user has confirmed the plate is cleared."""
  153. return printer_id in self._plate_cleared
  154. def consume_plate_cleared(self, printer_id: int):
  155. """Clear the plate-cleared flag (called when scheduler starts next print)."""
  156. self._plate_cleared.discard(printer_id)
  157. def set_event_loop(self, loop: asyncio.AbstractEventLoop):
  158. """Set the event loop for async callbacks."""
  159. self._loop = loop
  160. def set_print_start_callback(self, callback: Callable[[int, dict], None]):
  161. """Set callback for print start events."""
  162. self._on_print_start = callback
  163. def set_print_complete_callback(self, callback: Callable[[int, dict], None]):
  164. """Set callback for print completion events."""
  165. self._on_print_complete = callback
  166. def set_status_change_callback(self, callback: Callable[[int, PrinterState], None]):
  167. """Set callback for status change events."""
  168. self._on_status_change = callback
  169. def set_ams_change_callback(self, callback: Callable[[int, list], None]):
  170. """Set callback for AMS data change events."""
  171. self._on_ams_change = callback
  172. def set_layer_change_callback(self, callback: Callable[[int, int], None]):
  173. """Set callback for layer change events. Receives (printer_id, layer_num)."""
  174. self._on_layer_change = callback
  175. def set_bed_temp_update_callback(self, callback: Callable[[int, float], None]):
  176. """Set callback for bed temperature updates. Receives (printer_id, bed_temp)."""
  177. self._on_bed_temp_update = callback
  178. def _schedule_async(self, coro):
  179. """Schedule an async coroutine from a sync context.
  180. Captures exceptions from the coroutine and logs them to prevent
  181. silent failures in callbacks.
  182. """
  183. if self._loop and self._loop.is_running():
  184. future = asyncio.run_coroutine_threadsafe(coro, self._loop)
  185. def handle_exception(f):
  186. try:
  187. # This will re-raise any exception from the coroutine
  188. f.result()
  189. except Exception as e:
  190. import logging
  191. logging.getLogger(__name__).error(f"Exception in scheduled callback: {e}", exc_info=True)
  192. future.add_done_callback(handle_exception)
  193. async def connect_printer(self, printer: Printer) -> bool:
  194. """Connect to a printer."""
  195. if printer.id in self._clients:
  196. self.disconnect_printer(printer.id)
  197. printer_id = printer.id
  198. def on_state_change(state: PrinterState):
  199. if self._on_status_change:
  200. self._schedule_async(self._on_status_change(printer_id, state))
  201. def on_print_start(data: dict):
  202. if self._on_print_start:
  203. self._schedule_async(self._on_print_start(printer_id, data))
  204. def on_print_complete(data: dict):
  205. if self._on_print_complete:
  206. self._schedule_async(self._on_print_complete(printer_id, data))
  207. def on_ams_change(ams_data: list):
  208. if self._on_ams_change:
  209. self._schedule_async(self._on_ams_change(printer_id, ams_data))
  210. def on_layer_change(layer_num: int):
  211. if self._on_layer_change:
  212. self._schedule_async(self._on_layer_change(printer_id, layer_num))
  213. def on_bed_temp_update(bed_temp: float):
  214. if self._on_bed_temp_update:
  215. self._schedule_async(self._on_bed_temp_update(printer_id, bed_temp))
  216. client = BambuMQTTClient(
  217. ip_address=printer.ip_address,
  218. serial_number=printer.serial_number,
  219. access_code=printer.access_code,
  220. model=printer.model,
  221. on_state_change=on_state_change,
  222. on_print_start=on_print_start,
  223. on_print_complete=on_print_complete,
  224. on_ams_change=on_ams_change,
  225. on_layer_change=on_layer_change,
  226. on_bed_temp_update=on_bed_temp_update,
  227. )
  228. client.connect()
  229. self._clients[printer_id] = client
  230. self._models[printer_id] = printer.model # Cache model for feature detection
  231. self._printer_info[printer_id] = PrinterInfo(printer.name, printer.serial_number)
  232. # Wait a moment for connection
  233. await asyncio.sleep(1)
  234. return client.state.connected
  235. def disconnect_printer(self, printer_id: int, timeout: float = 0):
  236. """Disconnect from a printer."""
  237. if printer_id in self._clients:
  238. self._clients[printer_id].disconnect(timeout=timeout)
  239. del self._clients[printer_id]
  240. self._models.pop(printer_id, None) # Clean up model cache
  241. self._printer_info.pop(printer_id, None) # Clean up printer info cache
  242. def disconnect_all(self, timeout: float = 0):
  243. """Disconnect from all printers."""
  244. for printer_id in list(self._clients.keys()):
  245. self.disconnect_printer(printer_id, timeout=timeout)
  246. def get_status(self, printer_id: int) -> PrinterState | None:
  247. """Get the current status of a printer (checks for stale connections)."""
  248. if printer_id in self._clients:
  249. client = self._clients[printer_id]
  250. # Check staleness and update connected state if needed
  251. client.check_staleness()
  252. return client.state
  253. return None
  254. def get_model(self, printer_id: int) -> str | None:
  255. """Get the cached model for a printer."""
  256. return self._models.get(printer_id)
  257. def get_all_statuses(self) -> dict[int, PrinterState]:
  258. """Get status of all connected printers (checks for stale connections)."""
  259. result = {}
  260. for printer_id, client in self._clients.items():
  261. # Check staleness and update connected state if needed
  262. client.check_staleness()
  263. result[printer_id] = client.state
  264. return result
  265. def is_connected(self, printer_id: int) -> bool:
  266. """Check if a printer is connected (checks for stale connections)."""
  267. if printer_id in self._clients:
  268. client = self._clients[printer_id]
  269. # Check staleness and update connected state if needed
  270. return client.check_staleness()
  271. return False
  272. def get_client(self, printer_id: int) -> BambuMQTTClient | None:
  273. """Get the MQTT client for a printer."""
  274. return self._clients.get(printer_id)
  275. def mark_printer_offline(self, printer_id: int):
  276. """Mark a printer as offline and trigger status callback.
  277. This is used when we know the printer power was cut (e.g., smart plug turned off)
  278. to immediately update the UI without waiting for MQTT timeout.
  279. """
  280. import logging
  281. logger = logging.getLogger(__name__)
  282. if printer_id in self._clients:
  283. client = self._clients[printer_id]
  284. if client.state.connected:
  285. logger.info("Marking printer %s as offline (smart plug power off)", printer_id)
  286. client.state.connected = False
  287. client.state.state = "unknown"
  288. # Trigger the status change callback to broadcast via WebSocket
  289. if self._on_status_change:
  290. self._schedule_async(self._on_status_change(printer_id, client.state))
  291. def start_print(
  292. self,
  293. printer_id: int,
  294. filename: str,
  295. plate_id: int = 1,
  296. ams_mapping: list[int] | None = None,
  297. bed_levelling: bool = True,
  298. flow_cali: bool = False,
  299. vibration_cali: bool = True,
  300. layer_inspect: bool = False,
  301. timelapse: bool = False,
  302. use_ams: bool = True,
  303. ) -> bool:
  304. """Start a print on a connected printer."""
  305. caller = traceback.extract_stack(limit=3)[0]
  306. logger.info(
  307. "PRINT COMMAND: printer=%s, file=%s, caller=%s:%s:%s",
  308. printer_id,
  309. filename,
  310. caller.filename.split("/")[-1],
  311. caller.lineno,
  312. caller.name,
  313. )
  314. if printer_id in self._clients:
  315. return self._clients[printer_id].start_print(
  316. filename,
  317. plate_id,
  318. ams_mapping=ams_mapping,
  319. timelapse=timelapse,
  320. bed_levelling=bed_levelling,
  321. flow_cali=flow_cali,
  322. vibration_cali=vibration_cali,
  323. layer_inspect=layer_inspect,
  324. use_ams=use_ams,
  325. )
  326. return False
  327. def stop_print(self, printer_id: int) -> bool:
  328. """Stop the current print on a connected printer."""
  329. if printer_id in self._clients:
  330. return self._clients[printer_id].stop_print()
  331. return False
  332. async def wait_for_cooldown(
  333. self,
  334. printer_id: int,
  335. target_temp: float = 50.0,
  336. timeout: int = 600,
  337. check_interval: int = 10,
  338. ) -> bool:
  339. """Wait for the nozzle to cool down to a safe temperature.
  340. Args:
  341. printer_id: The printer to monitor
  342. target_temp: Target temperature to wait for (default 50°C)
  343. timeout: Maximum seconds to wait (default 600s = 10 min)
  344. check_interval: Seconds between temperature checks (default 10s)
  345. Returns:
  346. True if cooled down, False if timeout or not connected
  347. """
  348. import logging
  349. logger = logging.getLogger(__name__)
  350. elapsed = 0
  351. while elapsed < timeout:
  352. state = self.get_status(printer_id)
  353. if not state or not state.connected:
  354. logger.warning("Printer %s disconnected during cooldown wait", printer_id)
  355. return False
  356. # Check nozzle temperature (and nozzle_2 for dual extruders)
  357. nozzle_temp = state.temperatures.get("nozzle", 0)
  358. nozzle_2_temp = state.temperatures.get("nozzle_2", 0)
  359. max_temp = max(nozzle_temp, nozzle_2_temp)
  360. if max_temp <= target_temp:
  361. logger.info("Printer %s cooled down to %s°C", printer_id, max_temp)
  362. return True
  363. logger.debug("Printer %s nozzle at %s°C, waiting for %s°C...", printer_id, max_temp, target_temp)
  364. await asyncio.sleep(check_interval)
  365. elapsed += check_interval
  366. logger.warning("Printer %s cooldown timeout after %ss", printer_id, timeout)
  367. return False
  368. def enable_logging(self, printer_id: int, enabled: bool = True) -> bool:
  369. """Enable or disable MQTT logging for a printer."""
  370. if printer_id in self._clients:
  371. self._clients[printer_id].enable_logging(enabled)
  372. return True
  373. return False
  374. def get_logs(self, printer_id: int) -> list[MQTTLogEntry]:
  375. """Get MQTT logs for a printer."""
  376. if printer_id in self._clients:
  377. return self._clients[printer_id].get_logs()
  378. return []
  379. def clear_logs(self, printer_id: int) -> bool:
  380. """Clear MQTT logs for a printer."""
  381. if printer_id in self._clients:
  382. self._clients[printer_id].clear_logs()
  383. return True
  384. return False
  385. def is_logging_enabled(self, printer_id: int) -> bool:
  386. """Check if logging is enabled for a printer."""
  387. if printer_id in self._clients:
  388. return self._clients[printer_id].logging_enabled
  389. return False
  390. def send_drying_command(
  391. self,
  392. printer_id: int,
  393. ams_id: int,
  394. temp: int,
  395. duration: int,
  396. mode: int = 1,
  397. filament: str = "",
  398. rotate_tray: bool = False,
  399. ) -> bool:
  400. """Send AMS drying command to printer."""
  401. if printer_id not in self._clients:
  402. return False
  403. return self._clients[printer_id].send_drying_command(ams_id, temp, duration, mode, filament, rotate_tray)
  404. def request_status_update(self, printer_id: int) -> bool:
  405. """Request a full status update from the printer.
  406. This sends a 'pushall' command to get the latest data including nozzle info.
  407. """
  408. if printer_id in self._clients:
  409. return self._clients[printer_id].request_status_update()
  410. return False
  411. async def test_connection(
  412. self,
  413. ip_address: str,
  414. serial_number: str,
  415. access_code: str,
  416. ) -> dict:
  417. """Test connection to a printer without persisting."""
  418. client = BambuMQTTClient(
  419. ip_address=ip_address,
  420. serial_number=serial_number,
  421. access_code=access_code,
  422. )
  423. try:
  424. client.connect()
  425. await asyncio.sleep(2)
  426. result = {
  427. "success": client.state.connected,
  428. "state": client.state.state if client.state.connected else None,
  429. "model": client.state.raw_data.get("device_model"),
  430. }
  431. finally:
  432. client.disconnect()
  433. return result
  434. def get_derived_status_name(state: PrinterState, model: str | None = None) -> str | None:
  435. """
  436. Compute a human-readable status name based on printer state.
  437. Uses stg_cur when available, otherwise derives status from temperature data
  438. when the printer is heating before a print starts.
  439. Args:
  440. state: The printer state to analyze
  441. model: Optional printer model for model-specific workarounds
  442. """
  443. # Firmware bug: some models (A1, P1P, P1S) report stg_cur=0 when not printing.
  444. # stg_cur=0 maps to "Printing" in STAGE_NAMES, which incorrectly overrides the
  445. # real state (IDLE, FINISH, FAILED, etc.). Only trust stg_cur when the printer
  446. # is actually in an active print state (RUNNING or PAUSE).
  447. if state.state not in ("RUNNING", "PAUSE") and state.stg_cur == 0 and has_stg_cur_idle_bug(model):
  448. return None
  449. # If we have a valid calibration stage, use it
  450. # X1 models use -1 for idle, A1/P1 models use 255 for idle
  451. # Valid stage numbers are 0-254
  452. if 0 <= state.stg_cur < 255:
  453. return get_stage_name(state.stg_cur)
  454. # If not in RUNNING state, no derived status needed
  455. if state.state != "RUNNING":
  456. return None
  457. # Check if we're in an early phase where temperatures are heating
  458. temps = state.temperatures or {}
  459. progress = state.progress or 0
  460. # Only derive heating status when progress is very low (< 2%)
  461. # This indicates we're in the preparation phase, not actually printing
  462. if progress >= 2:
  463. return None
  464. # Check bed temperature - if target is set and current is significantly below
  465. bed_temp = temps.get("bed", 0)
  466. bed_target = temps.get("bed_target", 0)
  467. # Check nozzle temperature
  468. nozzle_temp = temps.get("nozzle", 0)
  469. nozzle_target = temps.get("nozzle_target", 0)
  470. # Temperature thresholds: consider "heating" if more than 10°C below target
  471. TEMP_THRESHOLD = 10
  472. # Determine what's heating (prioritize bed since it takes longer)
  473. if bed_target > 30 and (bed_target - bed_temp) > TEMP_THRESHOLD:
  474. return "Heating heatbed"
  475. elif nozzle_target > 30 and (nozzle_target - nozzle_temp) > TEMP_THRESHOLD:
  476. return "Heating nozzle"
  477. # If targets are set but we're close to them, we might be in final prep
  478. if bed_target > 30 or nozzle_target > 30:
  479. if progress == 0 and state.layer_num == 0:
  480. return "Preparing"
  481. return None
  482. def printer_state_to_dict(state: PrinterState, printer_id: int | None = None, model: str | None = None) -> dict:
  483. """Convert PrinterState to a JSON-serializable dict.
  484. Args:
  485. state: The printer state to convert
  486. printer_id: Optional printer ID for generating cover URLs
  487. model: Optional printer model for filtering unsupported features
  488. """
  489. # Parse AMS data from raw_data
  490. ams_units = []
  491. vt_tray = []
  492. raw_data = state.raw_data or {}
  493. # Build K-profile lookup map: cali_idx -> k_value
  494. kprofile_map: dict[int, float] = {}
  495. for kp in state.kprofiles or []:
  496. if kp.slot_id is not None and kp.k_value:
  497. try:
  498. kprofile_map[kp.slot_id] = float(kp.k_value)
  499. except (ValueError, TypeError):
  500. pass # Skip K-profile entries with unparseable values
  501. if "ams" in raw_data and isinstance(raw_data["ams"], list):
  502. for ams_data in raw_data["ams"]:
  503. trays = []
  504. for tray in ams_data.get("tray", []):
  505. tag_uid = tray.get("tag_uid")
  506. if tag_uid in ("", "0000000000000000"):
  507. tag_uid = None
  508. tray_uuid = tray.get("tray_uuid")
  509. if tray_uuid in ("", "00000000000000000000000000000000"):
  510. tray_uuid = None
  511. # Get K value: first try tray's k field, then lookup from K-profiles
  512. k_value = tray.get("k")
  513. cali_idx = tray.get("cali_idx")
  514. if k_value is None and cali_idx is not None and cali_idx in kprofile_map:
  515. k_value = kprofile_map[cali_idx]
  516. trays.append(
  517. {
  518. "id": int(tray.get("id", 0)),
  519. "tray_color": tray.get("tray_color"),
  520. "tray_type": tray.get("tray_type"),
  521. "tray_sub_brands": tray.get("tray_sub_brands"),
  522. "tray_id_name": tray.get("tray_id_name"),
  523. "tray_info_idx": tray.get("tray_info_idx"),
  524. "remain": tray.get("remain", 0),
  525. "k": k_value,
  526. "cali_idx": cali_idx,
  527. "tag_uid": tag_uid,
  528. "tray_uuid": tray_uuid,
  529. "nozzle_temp_min": tray.get("nozzle_temp_min"),
  530. "nozzle_temp_max": tray.get("nozzle_temp_max"),
  531. "drying_temp": tray.get("drying_temp"),
  532. "drying_time": tray.get("drying_time"),
  533. "state": tray.get("state"),
  534. }
  535. )
  536. # Prefer humidity_raw (actual percentage) over humidity (index 1-5)
  537. humidity_raw = ams_data.get("humidity_raw")
  538. humidity_idx = ams_data.get("humidity")
  539. humidity_value = None
  540. if humidity_raw is not None:
  541. try:
  542. humidity_value = int(humidity_raw)
  543. except (ValueError, TypeError):
  544. pass # Skip unparseable humidity; will try index fallback
  545. # Fall back to index if no raw value (index is 1-5, not percentage)
  546. if humidity_value is None and humidity_idx is not None:
  547. try:
  548. humidity_value = int(humidity_idx)
  549. except (ValueError, TypeError):
  550. pass # Skip unparseable humidity index; humidity remains None
  551. # AMS-HT has 1 tray, regular AMS has 4 trays
  552. is_ams_ht = len(trays) == 1
  553. ams_units.append(
  554. {
  555. "id": int(ams_data.get("id", 0)),
  556. "humidity": humidity_value,
  557. "temp": ams_data.get("temp"),
  558. "is_ams_ht": is_ams_ht,
  559. "tray": trays,
  560. # Serial number: Bambu MQTT uses "sn" key on AMS unit objects
  561. "serial_number": str(ams_data.get("sn") or ams_data.get("serial_number") or ""),
  562. # Firmware version: populated by _handle_version_info from get_version
  563. "sw_ver": str(ams_data.get("sw_ver") or ""),
  564. # Drying: dry_time > 0 means drying is active (minutes remaining)
  565. "dry_time": int(ams_data.get("dry_time") or 0),
  566. # Drying status from info hex bits (0=Off, 1=Checking, 2=Drying, 3=Cooling, etc.)
  567. "dry_status": int(ams_data.get("dry_status") or 0),
  568. "dry_sub_status": int(ams_data.get("dry_sub_status") or 0),
  569. # Cannot-dry reasons from firmware (e.g. 1=InsufficientPower, 8=NeedPluginPower)
  570. "dry_sf_reason": list(ams_data.get("dry_sf_reason") or []),
  571. # Module type: "ams", "n3f", "n3s" (from get_version)
  572. "module_type": str(ams_data.get("module_type") or ""),
  573. }
  574. )
  575. # Parse virtual tray (external spool) — now a list
  576. if "vt_tray" in raw_data:
  577. vt_tray_raw = raw_data["vt_tray"]
  578. # Defensive: MQTT sends vt_tray as a dict; normalize to list
  579. if isinstance(vt_tray_raw, dict):
  580. vt_tray_raw = [vt_tray_raw]
  581. elif not isinstance(vt_tray_raw, list):
  582. vt_tray_raw = []
  583. for vt_data in vt_tray_raw:
  584. vt_tag_uid = vt_data.get("tag_uid")
  585. if vt_tag_uid in ("", "0000000000000000"):
  586. vt_tag_uid = None
  587. vt_tray_uuid = vt_data.get("tray_uuid")
  588. if vt_tray_uuid in ("", "00000000000000000000000000000000"):
  589. vt_tray_uuid = None
  590. # Get K value for vt_tray
  591. vt_k_value = vt_data.get("k")
  592. vt_cali_idx = vt_data.get("cali_idx")
  593. if vt_k_value is None and vt_cali_idx is not None and vt_cali_idx in kprofile_map:
  594. vt_k_value = kprofile_map[vt_cali_idx]
  595. tray_id = int(vt_data.get("id", 254))
  596. vt_tray.append(
  597. {
  598. "id": tray_id,
  599. "tray_color": vt_data.get("tray_color"),
  600. "tray_type": vt_data.get("tray_type"),
  601. "tray_sub_brands": vt_data.get("tray_sub_brands"),
  602. "tray_id_name": vt_data.get("tray_id_name"),
  603. "tray_info_idx": vt_data.get("tray_info_idx"),
  604. "remain": vt_data.get("remain", 0),
  605. "k": vt_k_value,
  606. "cali_idx": vt_cali_idx,
  607. "tag_uid": vt_tag_uid,
  608. "tray_uuid": vt_tray_uuid,
  609. "nozzle_temp_min": vt_data.get("nozzle_temp_min"),
  610. "nozzle_temp_max": vt_data.get("nozzle_temp_max"),
  611. }
  612. )
  613. # Get ams_extruder_map from raw_data (populated by MQTT handler from AMS info field)
  614. ams_extruder_map = raw_data.get("ams_extruder_map", {})
  615. # Filter out chamber temp for models that don't have a real sensor
  616. # P1P, P1S, A1, A1Mini report meaningless chamber_temper values
  617. temperatures = state.temperatures
  618. if not supports_chamber_temp(model):
  619. temperatures = {
  620. k: v for k, v in temperatures.items() if k not in ("chamber", "chamber_target", "chamber_heating")
  621. }
  622. result = {
  623. "connected": state.connected,
  624. "state": state.state,
  625. "current_print": state.current_print,
  626. "subtask_name": state.subtask_name,
  627. "gcode_file": state.gcode_file,
  628. "progress": state.progress,
  629. "remaining_time": state.remaining_time,
  630. "layer_num": state.layer_num,
  631. "total_layers": state.total_layers,
  632. "temperatures": temperatures,
  633. "hms_errors": [
  634. {"code": e.code, "attr": e.attr, "module": e.module, "severity": e.severity}
  635. for e in (state.hms_errors or [])
  636. ],
  637. # AMS data for filament colors
  638. "ams": ams_units if ams_units else None,
  639. "vt_tray": vt_tray,
  640. # AMS status for filament change tracking
  641. "ams_status_main": state.ams_status_main,
  642. "ams_status_sub": state.ams_status_sub,
  643. "tray_now": state.tray_now,
  644. # Per-AMS extruder map: {ams_id: extruder_id} where 0=right, 1=left
  645. "ams_extruder_map": ams_extruder_map,
  646. # WiFi signal strength
  647. "wifi_signal": state.wifi_signal,
  648. "wired_network": state.wired_network,
  649. # Calibration stage tracking
  650. "stg_cur": state.stg_cur,
  651. "stg_cur_name": get_derived_status_name(state, model),
  652. # Printable objects count for skip objects feature
  653. "printable_objects_count": len(state.printable_objects),
  654. # Fan speeds (0-100 percentage, None if not available)
  655. "cooling_fan_speed": state.cooling_fan_speed,
  656. "big_fan1_speed": state.big_fan1_speed,
  657. "big_fan2_speed": state.big_fan2_speed,
  658. "heatbreak_fan_speed": state.heatbreak_fan_speed,
  659. # Chamber light state
  660. "chamber_light": state.chamber_light,
  661. # Active extruder for dual-nozzle printers (0=right, 1=left)
  662. "active_extruder": state.active_extruder,
  663. # H2C nozzle rack (tool-changer dock positions)
  664. # Map raw MQTT field names (type/diameter) to schema names (nozzle_type/nozzle_diameter)
  665. "nozzle_rack": [
  666. {
  667. "id": n.get("id", 0),
  668. "nozzle_type": n.get("type", ""),
  669. "nozzle_diameter": n.get("diameter", ""),
  670. "wear": n.get("wear"),
  671. "stat": n.get("stat"),
  672. "max_temp": n.get("max_temp", 0),
  673. "serial_number": n.get("serial_number", ""),
  674. "filament_color": n.get("filament_color", ""),
  675. "filament_id": n.get("filament_id", ""),
  676. }
  677. for n in (state.nozzle_rack or [])
  678. ],
  679. # AMS drying support
  680. "supports_drying": supports_drying(model, state.firmware_version),
  681. }
  682. # Add cover URL if there's an active print and printer_id is provided
  683. # Include PAUSE state so skip objects modal can show cover
  684. if printer_id and state.state in ("RUNNING", "PAUSE") and state.gcode_file:
  685. result["cover_url"] = f"/api/v1/printers/{printer_id}/cover"
  686. else:
  687. result["cover_url"] = None
  688. return result
  689. # Global printer manager instance
  690. printer_manager = PrinterManager()
  691. async def init_printer_connections(db: AsyncSession):
  692. """Initialize connections to all active printers."""
  693. result = await db.execute(select(Printer).where(Printer.is_active.is_(True)))
  694. printers = result.scalars().all()
  695. for printer in printers:
  696. await printer_manager.connect_printer(printer)