test_printer_manager.py 66 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712
  1. """Unit tests for PrinterManager service.
  2. Tests printer connection management, status tracking, and print control.
  3. """
  4. import logging
  5. from unittest.mock import AsyncMock, MagicMock, patch
  6. import pytest
  7. from backend.app.services.printer_manager import (
  8. PrinterManager,
  9. get_derived_status_name,
  10. has_stg_cur_idle_bug,
  11. init_printer_connections,
  12. parse_plate_id,
  13. printer_state_to_dict,
  14. supports_chamber_temp,
  15. supports_drying,
  16. )
  17. class TestPrinterManager:
  18. """Tests for PrinterManager class."""
  19. @pytest.fixture
  20. def manager(self):
  21. """Create a fresh PrinterManager instance."""
  22. return PrinterManager()
  23. @pytest.fixture
  24. def mock_printer(self):
  25. """Create a mock Printer object."""
  26. printer = MagicMock()
  27. printer.id = 1
  28. printer.ip_address = "192.168.1.100"
  29. printer.serial_number = "00M09A123456789"
  30. printer.access_code = "12345678"
  31. printer.is_active = True
  32. return printer
  33. @pytest.fixture
  34. def mock_client(self):
  35. """Create a mock BambuMQTTClient."""
  36. client = MagicMock()
  37. client.state = MagicMock()
  38. client.state.connected = True
  39. client.state.state = "IDLE"
  40. client.state.progress = 0
  41. client.state.temperatures = {"nozzle": 25, "bed": 25}
  42. client.state.raw_data = {}
  43. client.logging_enabled = False
  44. return client
  45. # ========================================================================
  46. # Tests for initialization
  47. # ========================================================================
  48. def test_init_creates_empty_clients_dict(self, manager):
  49. """Verify manager initializes with empty clients dict."""
  50. assert manager._clients == {}
  51. def test_init_callbacks_are_none(self, manager):
  52. """Verify all callbacks are initially None."""
  53. assert manager._on_print_start is None
  54. assert manager._on_print_complete is None
  55. assert manager._on_status_change is None
  56. assert manager._on_ams_change is None
  57. def test_init_loop_is_none(self, manager):
  58. """Verify event loop is initially None."""
  59. assert manager._loop is None
  60. # ========================================================================
  61. # Tests for callback setters
  62. # ========================================================================
  63. def test_set_event_loop(self, manager):
  64. """Verify event loop can be set."""
  65. mock_loop = MagicMock()
  66. manager.set_event_loop(mock_loop)
  67. assert manager._loop == mock_loop
  68. def test_set_print_start_callback(self, manager):
  69. """Verify print start callback can be set."""
  70. callback = MagicMock()
  71. manager.set_print_start_callback(callback)
  72. assert manager._on_print_start == callback
  73. def test_set_print_complete_callback(self, manager):
  74. """Verify print complete callback can be set."""
  75. callback = MagicMock()
  76. manager.set_print_complete_callback(callback)
  77. assert manager._on_print_complete == callback
  78. def test_set_status_change_callback(self, manager):
  79. """Verify status change callback can be set."""
  80. callback = MagicMock()
  81. manager.set_status_change_callback(callback)
  82. assert manager._on_status_change == callback
  83. def test_set_ams_change_callback(self, manager):
  84. """Verify AMS change callback can be set."""
  85. callback = MagicMock()
  86. manager.set_ams_change_callback(callback)
  87. assert manager._on_ams_change == callback
  88. # ========================================================================
  89. # Tests for _schedule_async
  90. # ========================================================================
  91. def test_schedule_async_with_running_loop(self, manager):
  92. """Verify async coroutine is scheduled when loop is running."""
  93. mock_loop = MagicMock()
  94. mock_loop.is_running.return_value = True
  95. manager._loop = mock_loop
  96. async def dummy_coro():
  97. pass
  98. coro = dummy_coro()
  99. manager._schedule_async(coro)
  100. mock_loop.is_running.assert_called_once()
  101. # Clean up the coroutine
  102. coro.close()
  103. def test_schedule_async_without_loop(self, manager):
  104. """Verify nothing happens when no loop is set."""
  105. async def dummy_coro():
  106. pass
  107. coro = dummy_coro()
  108. # Should not raise
  109. manager._schedule_async(coro)
  110. coro.close()
  111. def test_schedule_async_with_stopped_loop(self, manager):
  112. """Verify nothing happens when loop is not running."""
  113. mock_loop = MagicMock()
  114. mock_loop.is_running.return_value = False
  115. manager._loop = mock_loop
  116. async def dummy_coro():
  117. pass
  118. coro = dummy_coro()
  119. manager._schedule_async(coro)
  120. coro.close()
  121. # ========================================================================
  122. # Tests for connect_printer
  123. # ========================================================================
  124. @pytest.mark.asyncio
  125. async def test_connect_printer_creates_client(self, manager, mock_printer):
  126. """Verify connecting creates an MQTT client."""
  127. with patch("backend.app.services.printer_manager.BambuMQTTClient") as MockClient:
  128. mock_instance = MagicMock()
  129. mock_instance.state = MagicMock()
  130. mock_instance.state.connected = True
  131. MockClient.return_value = mock_instance
  132. result = await manager.connect_printer(mock_printer)
  133. MockClient.assert_called_once()
  134. mock_instance.connect.assert_called_once()
  135. assert mock_printer.id in manager._clients
  136. assert result is True
  137. @pytest.mark.asyncio
  138. async def test_connect_printer_disconnects_existing(self, manager, mock_printer, mock_client):
  139. """Verify connecting disconnects existing client first."""
  140. manager._clients[mock_printer.id] = mock_client
  141. with patch("backend.app.services.printer_manager.BambuMQTTClient") as MockClient:
  142. new_client = MagicMock()
  143. new_client.state = MagicMock()
  144. new_client.state.connected = True
  145. MockClient.return_value = new_client
  146. await manager.connect_printer(mock_printer)
  147. mock_client.disconnect.assert_called_once()
  148. @pytest.mark.asyncio
  149. async def test_connect_printer_returns_false_on_failure(self, manager, mock_printer):
  150. """Verify returns False when connection fails."""
  151. with patch("backend.app.services.printer_manager.BambuMQTTClient") as MockClient:
  152. mock_instance = MagicMock()
  153. mock_instance.state = MagicMock()
  154. mock_instance.state.connected = False
  155. MockClient.return_value = mock_instance
  156. result = await manager.connect_printer(mock_printer)
  157. assert result is False
  158. # ========================================================================
  159. # Tests for disconnect_printer
  160. # ========================================================================
  161. def test_disconnect_printer_removes_client(self, manager, mock_client):
  162. """Verify disconnecting removes and disconnects client."""
  163. manager._clients[1] = mock_client
  164. manager.disconnect_printer(1)
  165. mock_client.disconnect.assert_called_once()
  166. assert 1 not in manager._clients
  167. def test_disconnect_printer_handles_missing(self, manager):
  168. """Verify disconnecting non-existent printer doesn't raise."""
  169. manager.disconnect_printer(999) # Should not raise
  170. # ========================================================================
  171. # Tests for disconnect_all
  172. # ========================================================================
  173. def test_disconnect_all_disconnects_all_clients(self, manager):
  174. """Verify all clients are disconnected."""
  175. client1 = MagicMock()
  176. client2 = MagicMock()
  177. manager._clients[1] = client1
  178. manager._clients[2] = client2
  179. manager.disconnect_all()
  180. client1.disconnect.assert_called_once()
  181. client2.disconnect.assert_called_once()
  182. assert len(manager._clients) == 0
  183. # ========================================================================
  184. # Tests for get_status
  185. # ========================================================================
  186. def test_get_status_returns_state(self, manager, mock_client):
  187. """Verify get_status returns client state."""
  188. manager._clients[1] = mock_client
  189. result = manager.get_status(1)
  190. mock_client.check_staleness.assert_called_once()
  191. assert result == mock_client.state
  192. def test_get_status_returns_none_for_unknown(self, manager):
  193. """Verify get_status returns None for unknown printer."""
  194. result = manager.get_status(999)
  195. assert result is None
  196. # ========================================================================
  197. # Tests for get_all_statuses
  198. # ========================================================================
  199. def test_get_all_statuses_returns_all(self, manager):
  200. """Verify all statuses are returned."""
  201. client1 = MagicMock()
  202. client1.state = MagicMock(connected=True)
  203. client2 = MagicMock()
  204. client2.state = MagicMock(connected=False)
  205. manager._clients[1] = client1
  206. manager._clients[2] = client2
  207. result = manager.get_all_statuses()
  208. assert len(result) == 2
  209. assert 1 in result
  210. assert 2 in result
  211. client1.check_staleness.assert_called_once()
  212. client2.check_staleness.assert_called_once()
  213. # ========================================================================
  214. # Tests for is_connected
  215. # ========================================================================
  216. def test_is_connected_returns_true(self, manager, mock_client):
  217. """Verify is_connected returns True for connected printer."""
  218. mock_client.check_staleness.return_value = True
  219. manager._clients[1] = mock_client
  220. result = manager.is_connected(1)
  221. assert result is True
  222. def test_is_connected_returns_false_for_unknown(self, manager):
  223. """Verify is_connected returns False for unknown printer."""
  224. result = manager.is_connected(999)
  225. assert result is False
  226. # ========================================================================
  227. # Tests for get_client
  228. # ========================================================================
  229. def test_get_client_returns_client(self, manager, mock_client):
  230. """Verify get_client returns the client."""
  231. manager._clients[1] = mock_client
  232. result = manager.get_client(1)
  233. assert result == mock_client
  234. def test_get_client_returns_none_for_unknown(self, manager):
  235. """Verify get_client returns None for unknown printer."""
  236. result = manager.get_client(999)
  237. assert result is None
  238. # ========================================================================
  239. # Tests for mark_printer_offline
  240. # ========================================================================
  241. def test_mark_printer_offline_updates_state(self, manager, mock_client):
  242. """Verify mark_printer_offline updates client state."""
  243. mock_client.state.connected = True
  244. manager._clients[1] = mock_client
  245. manager.mark_printer_offline(1)
  246. assert mock_client.state.connected is False
  247. assert mock_client.state.state == "unknown"
  248. def test_mark_printer_offline_triggers_callback(self, manager, mock_client):
  249. """Verify mark_printer_offline triggers status callback."""
  250. mock_client.state.connected = True
  251. manager._clients[1] = mock_client
  252. # Callback must return a coroutine
  253. async def async_callback(printer_id, state):
  254. pass
  255. manager._on_status_change = async_callback
  256. # Need a running loop for callback
  257. mock_loop = MagicMock()
  258. mock_loop.is_running.return_value = True
  259. manager._loop = mock_loop
  260. manager.mark_printer_offline(1)
  261. # Callback should be scheduled via run_coroutine_threadsafe
  262. mock_loop.is_running.assert_called()
  263. # State should be updated
  264. assert mock_client.state.connected is False
  265. def test_mark_printer_offline_handles_unknown(self, manager):
  266. """Verify mark_printer_offline handles unknown printer."""
  267. manager.mark_printer_offline(999) # Should not raise
  268. def test_mark_printer_offline_skips_already_offline(self, manager, mock_client):
  269. """Verify mark_printer_offline skips already offline printer."""
  270. mock_client.state.connected = False
  271. manager._clients[1] = mock_client
  272. manager.mark_printer_offline(1)
  273. # State should remain unchanged
  274. assert mock_client.state.connected is False
  275. # ========================================================================
  276. # Tests for start_print
  277. # ========================================================================
  278. def test_start_print_calls_client(self, manager, mock_client):
  279. """Verify start_print calls client method."""
  280. mock_client.start_print.return_value = True
  281. manager._clients[1] = mock_client
  282. result = manager.start_print(1, "test.gcode")
  283. mock_client.start_print.assert_called_once_with(
  284. "test.gcode",
  285. 1,
  286. ams_mapping=None,
  287. timelapse=False,
  288. bed_levelling=True,
  289. flow_cali=False,
  290. vibration_cali=True,
  291. layer_inspect=False,
  292. use_ams=True,
  293. )
  294. assert result is True
  295. def test_start_print_returns_false_for_unknown(self, manager):
  296. """Verify start_print returns False for unknown printer."""
  297. result = manager.start_print(999, "test.gcode")
  298. assert result is False
  299. def test_start_print_logs_print_command_with_caller(self, manager, mock_client, caplog):
  300. """Verify start_print logs PRINT COMMAND with caller info (#374)."""
  301. mock_client.start_print.return_value = True
  302. manager._clients[1] = mock_client
  303. with caplog.at_level(logging.INFO, logger="backend.app.services.printer_manager"):
  304. manager.start_print(1, "benchy.3mf")
  305. print_cmd_logs = [r for r in caplog.records if "PRINT COMMAND" in r.message]
  306. assert len(print_cmd_logs) == 1
  307. log_msg = print_cmd_logs[0].message
  308. assert "printer=1" in log_msg
  309. assert "file=benchy.3mf" in log_msg
  310. assert "caller=" in log_msg
  311. def test_start_print_logs_even_when_printer_unknown(self, manager, caplog):
  312. """Verify PRINT COMMAND is logged even for unknown printers (#374)."""
  313. with caplog.at_level(logging.INFO, logger="backend.app.services.printer_manager"):
  314. result = manager.start_print(999, "ghost.3mf")
  315. assert result is False
  316. print_cmd_logs = [r for r in caplog.records if "PRINT COMMAND" in r.message]
  317. assert len(print_cmd_logs) == 1
  318. # ========================================================================
  319. # Tests for stop_print
  320. # ========================================================================
  321. def test_stop_print_calls_client(self, manager, mock_client):
  322. """Verify stop_print calls client method."""
  323. mock_client.stop_print.return_value = True
  324. manager._clients[1] = mock_client
  325. result = manager.stop_print(1)
  326. mock_client.stop_print.assert_called_once()
  327. assert result is True
  328. def test_stop_print_returns_false_for_unknown(self, manager):
  329. """Verify stop_print returns False for unknown printer."""
  330. result = manager.stop_print(999)
  331. assert result is False
  332. # ========================================================================
  333. # Tests for wait_for_cooldown
  334. # ========================================================================
  335. @pytest.mark.asyncio
  336. async def test_wait_for_cooldown_returns_true_when_cool(self, manager, mock_client):
  337. """Verify wait_for_cooldown returns True when printer is cool."""
  338. mock_client.state.connected = True
  339. mock_client.state.temperatures = {"nozzle": 40, "bed": 30}
  340. mock_client.check_staleness.return_value = True
  341. manager._clients[1] = mock_client
  342. result = await manager.wait_for_cooldown(1, target_temp=50)
  343. assert result is True
  344. @pytest.mark.asyncio
  345. async def test_wait_for_cooldown_returns_false_on_disconnect(self, manager, mock_client):
  346. """Verify wait_for_cooldown returns False when printer disconnects."""
  347. mock_client.state.connected = False
  348. mock_client.check_staleness.return_value = False
  349. manager._clients[1] = mock_client
  350. result = await manager.wait_for_cooldown(1, target_temp=50, timeout=1)
  351. assert result is False
  352. @pytest.mark.asyncio
  353. async def test_wait_for_cooldown_returns_false_for_unknown(self, manager):
  354. """Verify wait_for_cooldown returns False for unknown printer."""
  355. result = await manager.wait_for_cooldown(999, target_temp=50, timeout=1)
  356. assert result is False
  357. @pytest.mark.asyncio
  358. async def test_wait_for_cooldown_checks_both_nozzles(self, manager, mock_client):
  359. """Verify wait_for_cooldown checks both nozzles for dual extruders."""
  360. mock_client.state.connected = True
  361. mock_client.state.temperatures = {"nozzle": 40, "nozzle_2": 45, "bed": 30}
  362. mock_client.check_staleness.return_value = True
  363. manager._clients[1] = mock_client
  364. result = await manager.wait_for_cooldown(1, target_temp=50)
  365. assert result is True
  366. # ========================================================================
  367. # Tests for logging methods
  368. # ========================================================================
  369. def test_enable_logging_calls_client(self, manager, mock_client):
  370. """Verify enable_logging calls client method."""
  371. manager._clients[1] = mock_client
  372. result = manager.enable_logging(1, True)
  373. mock_client.enable_logging.assert_called_once_with(True)
  374. assert result is True
  375. def test_enable_logging_returns_false_for_unknown(self, manager):
  376. """Verify enable_logging returns False for unknown printer."""
  377. result = manager.enable_logging(999, True)
  378. assert result is False
  379. def test_get_logs_returns_logs(self, manager, mock_client):
  380. """Verify get_logs returns client logs."""
  381. mock_logs = [MagicMock(), MagicMock()]
  382. mock_client.get_logs.return_value = mock_logs
  383. manager._clients[1] = mock_client
  384. result = manager.get_logs(1)
  385. assert result == mock_logs
  386. def test_get_logs_returns_empty_for_unknown(self, manager):
  387. """Verify get_logs returns empty list for unknown printer."""
  388. result = manager.get_logs(999)
  389. assert result == []
  390. def test_clear_logs_calls_client(self, manager, mock_client):
  391. """Verify clear_logs calls client method."""
  392. manager._clients[1] = mock_client
  393. result = manager.clear_logs(1)
  394. mock_client.clear_logs.assert_called_once()
  395. assert result is True
  396. def test_clear_logs_returns_false_for_unknown(self, manager):
  397. """Verify clear_logs returns False for unknown printer."""
  398. result = manager.clear_logs(999)
  399. assert result is False
  400. def test_is_logging_enabled_returns_status(self, manager, mock_client):
  401. """Verify is_logging_enabled returns client status."""
  402. mock_client.logging_enabled = True
  403. manager._clients[1] = mock_client
  404. result = manager.is_logging_enabled(1)
  405. assert result is True
  406. def test_is_logging_enabled_returns_false_for_unknown(self, manager):
  407. """Verify is_logging_enabled returns False for unknown printer."""
  408. result = manager.is_logging_enabled(999)
  409. assert result is False
  410. # ========================================================================
  411. # Tests for request_status_update
  412. # ========================================================================
  413. def test_request_status_update_calls_client(self, manager, mock_client):
  414. """Verify request_status_update calls client method."""
  415. mock_client.request_status_update.return_value = True
  416. manager._clients[1] = mock_client
  417. result = manager.request_status_update(1)
  418. mock_client.request_status_update.assert_called_once()
  419. assert result is True
  420. def test_request_status_update_returns_false_for_unknown(self, manager):
  421. """Verify request_status_update returns False for unknown printer."""
  422. result = manager.request_status_update(999)
  423. assert result is False
  424. # ========================================================================
  425. # Tests for test_connection
  426. # ========================================================================
  427. @pytest.mark.asyncio
  428. async def test_test_connection_success(self, manager):
  429. """Verify test_connection returns success on connection."""
  430. with patch("backend.app.services.printer_manager.BambuMQTTClient") as MockClient:
  431. mock_instance = MagicMock()
  432. mock_instance.state = MagicMock()
  433. mock_instance.state.connected = True
  434. mock_instance.state.state = "IDLE"
  435. mock_instance.state.raw_data = {"device_model": "X1C"}
  436. MockClient.return_value = mock_instance
  437. result = await manager.test_connection("192.168.1.100", "00M09A123456789", "12345678")
  438. assert result["success"] is True
  439. assert result["state"] == "IDLE"
  440. assert result["model"] == "X1C"
  441. mock_instance.disconnect.assert_called_once()
  442. @pytest.mark.asyncio
  443. async def test_test_connection_failure(self, manager):
  444. """Verify test_connection returns failure on connection error."""
  445. with patch("backend.app.services.printer_manager.BambuMQTTClient") as MockClient:
  446. mock_instance = MagicMock()
  447. mock_instance.state = MagicMock()
  448. mock_instance.state.connected = False
  449. MockClient.return_value = mock_instance
  450. # Shorten the probe budget so the test doesn't burn the full
  451. # 8-second production timeout while polling a failing connection.
  452. with (
  453. patch.object(manager, "PROBE_TIMEOUT_SECONDS", 0.4),
  454. patch.object(manager, "PROBE_POLL_INTERVAL_SECONDS", 0.1),
  455. ):
  456. result = await manager.test_connection("192.168.1.100", "00M09A123456789", "12345678")
  457. assert result["success"] is False
  458. assert result["state"] is None
  459. mock_instance.disconnect.assert_called_once()
  460. @pytest.mark.asyncio
  461. async def test_test_connection_polls_and_returns_early_on_connect(self, manager):
  462. """#1445: a slow printer that finishes its handshake mid-probe must
  463. not be reported as a failure. Previously a fixed 2s sleep meant P1S
  464. TLS / CONNACK that took 3-5s got falsely rejected; now we poll and
  465. early-return as soon as connected flips True.
  466. """
  467. import asyncio
  468. import time
  469. with patch("backend.app.services.printer_manager.BambuMQTTClient") as MockClient:
  470. mock_instance = MagicMock()
  471. mock_instance.state = MagicMock()
  472. mock_instance.state.connected = False # not connected at probe start
  473. mock_instance.state.state = "IDLE"
  474. mock_instance.state.raw_data = {"device_model": "P1S"}
  475. MockClient.return_value = mock_instance
  476. async def flip_connected_after(delay: float):
  477. await asyncio.sleep(delay)
  478. mock_instance.state.connected = True
  479. # Simulates the P1S broker finishing its slow handshake ~0.5s in,
  480. # well past the old 2s-or-fail boundary's natural variance.
  481. with (
  482. patch.object(manager, "PROBE_TIMEOUT_SECONDS", 3.0),
  483. patch.object(manager, "PROBE_POLL_INTERVAL_SECONDS", 0.05),
  484. ):
  485. start = time.monotonic()
  486. flip_task = asyncio.create_task(flip_connected_after(0.5))
  487. try:
  488. result = await manager.test_connection("192.168.1.100", "00M09A123456789", "12345678")
  489. finally:
  490. await flip_task
  491. elapsed = time.monotonic() - start
  492. assert result["success"] is True
  493. assert result["state"] == "IDLE"
  494. # Early-return guarantee: must come back well before the configured
  495. # timeout once connected flips. ~0.5s + one poll interval is plenty.
  496. assert elapsed < 1.5, f"probe should have early-returned shortly after 0.5s, took {elapsed:.2f}s"
  497. @pytest.mark.asyncio
  498. async def test_test_connection_disconnect_runs_off_loop(self, manager):
  499. """#1445: the root cause of the "Docker container hangs" symptom was
  500. `client.disconnect()` running on the asyncio thread — paho's
  501. `loop_stop()` does a thread-join that blocks until its network
  502. thread exits, which on a slow P1S TLS handshake could take many
  503. seconds. This test pins the off-loop teardown so a regression that
  504. reintroduces sync disconnect breaks CI immediately.
  505. """
  506. import asyncio
  507. import threading
  508. import time
  509. with patch("backend.app.services.printer_manager.BambuMQTTClient") as MockClient:
  510. asyncio_thread_id = threading.get_ident()
  511. disconnect_thread_ids: list[int] = []
  512. disconnect_blocked_for: list[float] = []
  513. def slow_blocking_disconnect():
  514. # Mirrors paho.Client.loop_stop()'s thread-join semantics —
  515. # if this runs on the asyncio thread the event loop stalls.
  516. disconnect_thread_ids.append(threading.get_ident())
  517. start = time.monotonic()
  518. time.sleep(0.4)
  519. disconnect_blocked_for.append(time.monotonic() - start)
  520. mock_instance = MagicMock()
  521. mock_instance.state = MagicMock()
  522. mock_instance.state.connected = True
  523. mock_instance.state.state = "IDLE"
  524. mock_instance.state.raw_data = {"device_model": "P1S"}
  525. mock_instance.disconnect = slow_blocking_disconnect
  526. MockClient.return_value = mock_instance
  527. # Another coroutine must keep making progress while disconnect()
  528. # runs — proves the event loop was not blocked.
  529. event_loop_alive_ticks = 0
  530. async def heartbeat():
  531. nonlocal event_loop_alive_ticks
  532. while True:
  533. await asyncio.sleep(0.05)
  534. event_loop_alive_ticks += 1
  535. heartbeat_task = asyncio.create_task(heartbeat())
  536. try:
  537. await manager.test_connection("192.168.1.100", "00M09A123456789", "12345678")
  538. finally:
  539. heartbeat_task.cancel()
  540. try:
  541. await heartbeat_task
  542. except asyncio.CancelledError:
  543. pass
  544. # disconnect ran on a different thread than asyncio's
  545. assert disconnect_thread_ids, "disconnect was never called"
  546. assert disconnect_thread_ids[0] != asyncio_thread_id, (
  547. "disconnect ran on the asyncio thread — this blocks the event loop (#1445)"
  548. )
  549. # Heartbeat made progress while the 0.4s disconnect was blocking
  550. # the worker thread (proves the loop wasn't stalled).
  551. assert event_loop_alive_ticks >= 3, (
  552. f"event loop appears to have stalled during disconnect "
  553. f"(only {event_loop_alive_ticks} heartbeats; expected >=3)"
  554. )
  555. # ========================================================================
  556. # Tests for current print user tracking (Issue #206)
  557. # ========================================================================
  558. def test_set_current_print_user(self, manager):
  559. """Verify current print user can be set."""
  560. manager.set_current_print_user(1, 42, "testuser")
  561. assert 1 in manager._current_print_user
  562. assert manager._current_print_user[1]["user_id"] == 42
  563. assert manager._current_print_user[1]["username"] == "testuser"
  564. def test_get_current_print_user_returns_user(self, manager):
  565. """Verify get_current_print_user returns the stored user."""
  566. manager.set_current_print_user(1, 42, "testuser")
  567. result = manager.get_current_print_user(1)
  568. assert result is not None
  569. assert result["user_id"] == 42
  570. assert result["username"] == "testuser"
  571. def test_get_current_print_user_returns_none_for_unknown(self, manager):
  572. """Verify get_current_print_user returns None for unknown printer."""
  573. result = manager.get_current_print_user(999)
  574. assert result is None
  575. def test_clear_current_print_user(self, manager):
  576. """Verify current print user can be cleared."""
  577. manager.set_current_print_user(1, 42, "testuser")
  578. manager.clear_current_print_user(1)
  579. result = manager.get_current_print_user(1)
  580. assert result is None
  581. def test_clear_current_print_user_no_error_for_unknown(self, manager):
  582. """Verify clearing unknown printer doesn't raise error."""
  583. # Should not raise
  584. manager.clear_current_print_user(999)
  585. def test_set_current_print_user_overwrites_existing(self, manager):
  586. """Verify setting user overwrites existing value."""
  587. manager.set_current_print_user(1, 42, "user1")
  588. manager.set_current_print_user(1, 99, "user2")
  589. result = manager.get_current_print_user(1)
  590. assert result["user_id"] == 99
  591. assert result["username"] == "user2"
  592. def test_multiple_printers_have_separate_users(self, manager):
  593. """Verify each printer tracks its own user separately."""
  594. manager.set_current_print_user(1, 42, "user1")
  595. manager.set_current_print_user(2, 99, "user2")
  596. result1 = manager.get_current_print_user(1)
  597. result2 = manager.get_current_print_user(2)
  598. assert result1["username"] == "user1"
  599. assert result2["username"] == "user2"
  600. class TestPrinterStateToDict:
  601. """Tests for printer_state_to_dict helper function."""
  602. @pytest.fixture
  603. def mock_state(self):
  604. """Create a mock PrinterState."""
  605. state = MagicMock()
  606. state.connected = True
  607. state.state = "RUNNING"
  608. state.current_print = "test.3mf"
  609. state.subtask_name = "Test Print"
  610. state.gcode_file = "/sdcard/test.gcode"
  611. state.progress = 50
  612. state.remaining_time = 3600
  613. state.layer_num = 10
  614. state.total_layers = 20
  615. state.temperatures = {"nozzle": 200, "bed": 60}
  616. state.hms_errors = []
  617. state.ams_status_main = 0
  618. state.ams_status_sub = 0
  619. state.tray_now = "1"
  620. state.wifi_signal = -50
  621. state.raw_data = {}
  622. state.stg_cur = -1 # No calibration stage active
  623. state.firmware_version = None
  624. return state
  625. def test_basic_conversion(self, mock_state):
  626. """Verify basic state fields are converted."""
  627. result = printer_state_to_dict(mock_state)
  628. assert result["connected"] is True
  629. assert result["state"] == "RUNNING"
  630. assert result["progress"] == 50
  631. assert result["temperatures"] == {"nozzle": 200, "bed": 60}
  632. def test_ams_data_parsing(self, mock_state):
  633. """Verify AMS data is parsed correctly."""
  634. mock_state.raw_data = {
  635. "ams": [
  636. {
  637. "id": 0,
  638. "humidity_raw": 45,
  639. "temp": 25,
  640. "tray": [
  641. {
  642. "id": 0,
  643. "tray_color": "FF0000",
  644. "tray_type": "PLA",
  645. "tray_sub_brands": "Generic",
  646. "remain": 80,
  647. "k": 0.5,
  648. "tag_uid": "ABC123",
  649. "tray_uuid": "uuid-123",
  650. }
  651. ],
  652. }
  653. ]
  654. }
  655. result = printer_state_to_dict(mock_state)
  656. assert result["ams"] is not None
  657. assert len(result["ams"]) == 1
  658. assert result["ams"][0]["humidity"] == 45
  659. assert len(result["ams"][0]["tray"]) == 1
  660. assert result["ams"][0]["tray"][0]["tray_color"] == "FF0000"
  661. def test_empty_tag_uid_becomes_none(self, mock_state):
  662. """Verify empty tag_uid is converted to None."""
  663. mock_state.raw_data = {
  664. "ams": [
  665. {
  666. "id": 0,
  667. "tray": [
  668. {
  669. "id": 0,
  670. "tag_uid": "",
  671. "tray_uuid": "00000000000000000000000000000000",
  672. }
  673. ],
  674. }
  675. ]
  676. }
  677. result = printer_state_to_dict(mock_state)
  678. assert result["ams"][0]["tray"][0]["tag_uid"] is None
  679. assert result["ams"][0]["tray"][0]["tray_uuid"] is None
  680. def test_bare_tray_emulates_state_9(self, mock_state):
  681. """P1S / A1 Mini physically-empty-slot signal (#1322 follow-up by @RosdasHH):
  682. the firmware sends only `{"id": N}` for a truly empty slot. Treat that as
  683. the firmware's "no spool" state (state=9) so the inventory assign-spool
  684. path can short-circuit the doomed MQTT publish.
  685. """
  686. mock_state.raw_data = {
  687. "ams": [
  688. {
  689. "id": 0,
  690. "tray": [
  691. {"id": 0, "state": 11, "tray_type": "PLA"}, # loaded slot
  692. {"id": 1}, # P1S empty-slot signal — only id
  693. ],
  694. }
  695. ]
  696. }
  697. result = printer_state_to_dict(mock_state)
  698. trays = result["ams"][0]["tray"]
  699. assert trays[0]["state"] == 11, "loaded slot keeps its firmware state"
  700. assert trays[1]["state"] == 9, "bare {id} tray must be promoted to state=9"
  701. def test_populated_payload_with_empty_state_3_is_not_promoted(self, mock_state):
  702. """A1 Mini BMCU / P1S Standard AMS post-Reset-Slot case (#1322 root):
  703. firmware sends state=3 + tray_type="" but with the FULL field set
  704. populated. Must NOT be confused with the bare-tray empty signal —
  705. else inventory.py would short-circuit MQTT and we'd reintroduce the
  706. deadlock the #1322 fix removed.
  707. """
  708. mock_state.raw_data = {
  709. "ams": [
  710. {
  711. "id": 0,
  712. "tray": [
  713. {
  714. "id": 0,
  715. "state": 3,
  716. "tray_type": "", # cleared
  717. "tray_color": "",
  718. "tag_uid": "0000000000000000",
  719. "remain": 0,
  720. }
  721. ],
  722. }
  723. ]
  724. }
  725. result = printer_state_to_dict(mock_state)
  726. # state stays at 3 — the bare-tray promotion requires the dict to have
  727. # ONLY the id key, not just empty/falsy values for the other fields.
  728. assert result["ams"][0]["tray"][0]["state"] == 3
  729. def test_zero_tag_uid_becomes_none(self, mock_state):
  730. """Verify zero tag_uid is converted to None."""
  731. mock_state.raw_data = {
  732. "ams": [
  733. {
  734. "id": 0,
  735. "tray": [
  736. {
  737. "id": 0,
  738. "tag_uid": "0000000000000000",
  739. }
  740. ],
  741. }
  742. ]
  743. }
  744. result = printer_state_to_dict(mock_state)
  745. assert result["ams"][0]["tray"][0]["tag_uid"] is None
  746. def test_vt_tray_parsing(self, mock_state):
  747. """Verify virtual tray is parsed correctly as a list."""
  748. mock_state.raw_data = {
  749. "vt_tray": [
  750. {
  751. "tray_color": "00FF00",
  752. "tray_type": "PETG",
  753. "tray_sub_brands": "Generic",
  754. "remain": 60,
  755. "tag_uid": "VT123",
  756. }
  757. ]
  758. }
  759. result = printer_state_to_dict(mock_state)
  760. assert isinstance(result["vt_tray"], list)
  761. assert len(result["vt_tray"]) == 1
  762. assert result["vt_tray"][0]["id"] == 254
  763. assert result["vt_tray"][0]["tray_color"] == "00FF00"
  764. assert result["vt_tray"][0]["tray_type"] == "PETG"
  765. def test_vt_tray_dict_normalized_to_list(self, mock_state):
  766. """Verify vt_tray as a raw dict (from MQTT) is normalized to a list."""
  767. mock_state.raw_data = {
  768. "vt_tray": {
  769. "id": "254",
  770. "tray_color": "FF0000",
  771. "tray_type": "PLA",
  772. "tray_sub_brands": "Generic",
  773. "tag_uid": "0000000000000000",
  774. "tray_uuid": "00000000000000000000000000000000",
  775. "remain": 0,
  776. }
  777. }
  778. result = printer_state_to_dict(mock_state)
  779. assert isinstance(result["vt_tray"], list)
  780. assert len(result["vt_tray"]) == 1
  781. assert result["vt_tray"][0]["tray_color"] == "FF0000"
  782. assert result["vt_tray"][0]["tray_type"] == "PLA"
  783. assert result["vt_tray"][0]["tag_uid"] is None
  784. assert result["vt_tray"][0]["tray_uuid"] is None
  785. def test_vt_tray_non_list_non_dict_ignored(self, mock_state):
  786. """Verify unexpected vt_tray types (e.g. string) produce empty list."""
  787. mock_state.raw_data = {"vt_tray": "unexpected_string"}
  788. result = printer_state_to_dict(mock_state)
  789. assert result["vt_tray"] == []
  790. def test_hms_errors_conversion(self, mock_state):
  791. """Verify HMS errors are converted correctly."""
  792. error = MagicMock()
  793. error.code = "0700_0100"
  794. error.attr = 1
  795. error.module = "AMS"
  796. error.severity = 2
  797. mock_state.hms_errors = [error]
  798. result = printer_state_to_dict(mock_state)
  799. assert len(result["hms_errors"]) == 1
  800. assert result["hms_errors"][0]["code"] == "0700_0100"
  801. assert result["hms_errors"][0]["module"] == "AMS"
  802. def test_cover_url_added_for_running_print(self, mock_state):
  803. """Verify cover_url is added for running prints."""
  804. result = printer_state_to_dict(mock_state, printer_id=1)
  805. assert result["cover_url"] == "/api/v1/printers/1/cover"
  806. def test_current_plate_id_extracted_from_gcode_file(self, mock_state):
  807. """Verify current_plate_id is parsed from a Bambu plate path (#881)."""
  808. mock_state.gcode_file = "/Metadata/plate_3.gcode"
  809. result = printer_state_to_dict(mock_state)
  810. assert result["current_plate_id"] == 3
  811. def test_current_plate_id_none_when_no_plate_segment(self, mock_state):
  812. """Verify current_plate_id stays None when gcode_file has no plate marker."""
  813. mock_state.gcode_file = "/sdcard/test.gcode"
  814. result = printer_state_to_dict(mock_state)
  815. assert result["current_plate_id"] is None
  816. def test_cover_url_none_when_not_running(self, mock_state):
  817. """Verify cover_url is None when not printing."""
  818. mock_state.state = "IDLE"
  819. result = printer_state_to_dict(mock_state, printer_id=1)
  820. assert result["cover_url"] is None
  821. def test_ams_ht_detection(self, mock_state):
  822. """Verify AMS-HT is detected (1 tray vs 4)."""
  823. mock_state.raw_data = {
  824. "ams": [
  825. {
  826. "id": 0,
  827. "tray": [{"id": 0}], # Only 1 tray = AMS-HT
  828. }
  829. ]
  830. }
  831. result = printer_state_to_dict(mock_state)
  832. assert result["ams"][0]["is_ams_ht"] is True
  833. def test_regular_ams_detection(self, mock_state):
  834. """Verify regular AMS is detected (4 trays)."""
  835. mock_state.raw_data = {"ams": [{"id": 0, "tray": [{"id": 0}, {"id": 1}, {"id": 2}, {"id": 3}]}]}
  836. result = printer_state_to_dict(mock_state)
  837. assert result["ams"][0]["is_ams_ht"] is False
  838. def test_chamber_temp_filtered_for_p1s(self, mock_state):
  839. """Verify chamber temperature is filtered out for P1S (no chamber sensor)."""
  840. mock_state.temperatures = {
  841. "nozzle": 200,
  842. "bed": 60,
  843. "chamber": 5,
  844. "chamber_target": 0,
  845. "chamber_heating": False,
  846. }
  847. result = printer_state_to_dict(mock_state, model="P1S")
  848. assert "chamber" not in result["temperatures"]
  849. assert "chamber_target" not in result["temperatures"]
  850. assert "chamber_heating" not in result["temperatures"]
  851. assert result["temperatures"]["nozzle"] == 200
  852. assert result["temperatures"]["bed"] == 60
  853. def test_chamber_temp_kept_for_x1c(self, mock_state):
  854. """Verify chamber temperature is kept for X1C (has chamber sensor)."""
  855. mock_state.temperatures = {
  856. "nozzle": 200,
  857. "bed": 60,
  858. "chamber": 25,
  859. "chamber_target": 45,
  860. "chamber_heating": True,
  861. }
  862. result = printer_state_to_dict(mock_state, model="X1C")
  863. assert result["temperatures"]["chamber"] == 25
  864. assert result["temperatures"]["chamber_target"] == 45
  865. assert result["temperatures"]["chamber_heating"] is True
  866. def test_chamber_temp_filtered_for_a1(self, mock_state):
  867. """Verify chamber temperature is filtered out for A1 (no chamber sensor)."""
  868. mock_state.temperatures = {"nozzle": 200, "bed": 60, "chamber": 5}
  869. result = printer_state_to_dict(mock_state, model="A1")
  870. assert "chamber" not in result["temperatures"]
  871. def test_chamber_temp_kept_when_no_model(self, mock_state):
  872. """Verify chamber temperature is kept when model is not specified (conservative approach)."""
  873. mock_state.temperatures = {"nozzle": 200, "bed": 60, "chamber": 25}
  874. result = printer_state_to_dict(mock_state) # No model specified
  875. # When model is unknown, we can't filter - leave as is
  876. # Actually supports_chamber_temp returns False for None, so it will filter
  877. # Let's check the actual behavior
  878. assert "chamber" not in result["temperatures"]
  879. def test_ams_drying_fields_included(self, mock_state):
  880. """Verify AMS drying fields (dry_time, module_type) are included in output."""
  881. mock_state.raw_data = {
  882. "ams": [
  883. {
  884. "id": 0,
  885. "dry_time": 42,
  886. "module_type": "n3f",
  887. "tray": [
  888. {
  889. "id": 0,
  890. "tray_color": "FF0000",
  891. "tray_type": "PLA",
  892. "drying_temp": 55,
  893. "drying_time": 240,
  894. }
  895. ],
  896. }
  897. ]
  898. }
  899. result = printer_state_to_dict(mock_state)
  900. ams_unit = result["ams"][0]
  901. assert ams_unit["dry_time"] == 42
  902. assert ams_unit["module_type"] == "n3f"
  903. # Tray-level drying fields
  904. tray = ams_unit["tray"][0]
  905. assert tray["drying_temp"] == 55
  906. assert tray["drying_time"] == 240
  907. def test_awaiting_plate_clear_defaults_false(self, mock_state):
  908. """Without a printer_id, awaiting_plate_clear is False (no lookup possible)."""
  909. result = printer_state_to_dict(mock_state)
  910. assert result["awaiting_plate_clear"] is False
  911. def test_awaiting_plate_clear_surfaced_when_set(self, mock_state):
  912. """With printer_id, awaiting_plate_clear reflects PrinterManager state.
  913. Regression: PR #939 left this flag off the WebSocket payload, so the
  914. "Clear Plate" button only appeared after the 30 s REST fallback poll.
  915. """
  916. from backend.app.services.printer_manager import printer_manager
  917. printer_manager.set_awaiting_plate_clear(12345, True)
  918. try:
  919. result = printer_state_to_dict(mock_state, printer_id=12345)
  920. assert result["awaiting_plate_clear"] is True
  921. finally:
  922. printer_manager.set_awaiting_plate_clear(12345, False)
  923. def test_name_and_model_surfaced_when_registered(self, mock_state):
  924. """Registered PrinterInfo name + model arg should land in the WS payload.
  925. Regression for #963 follow-up: without this, the gcode viewer's printer
  926. selector had to wait on a /printers fetch before it could render real
  927. names, and the initial WS snapshot showed "Printer 1" fallbacks.
  928. """
  929. from backend.app.services.printer_manager import PrinterInfo, printer_manager
  930. # Register a stub PrinterInfo; the real manager writes this on connect.
  931. printer_manager._printer_info[98765] = PrinterInfo(name="My X1C", serial_number="01S00-0")
  932. try:
  933. result = printer_state_to_dict(mock_state, printer_id=98765, model="X1C")
  934. assert result["name"] == "My X1C"
  935. assert result["model"] == "X1C"
  936. finally:
  937. printer_manager._printer_info.pop(98765, None)
  938. def test_name_and_model_absent_when_no_printer_id(self, mock_state):
  939. """Without a printer_id (unusual callsites), name/model keys stay absent.
  940. The consumers (gcode viewer, frontend card) tolerate missing keys; what
  941. they can't tolerate is an unrelated printer's name accidentally leaking
  942. into a status meant for a different one.
  943. """
  944. result = printer_state_to_dict(mock_state)
  945. assert "name" not in result
  946. assert "model" not in result
  947. def test_model_absent_when_arg_is_none(self, mock_state):
  948. """`model` arg=None must not plant a `model` key at all.
  949. If the arg is None, callers didn't know the model yet; emitting a
  950. `model: null` field would overwrite a good value cached client-side.
  951. """
  952. from backend.app.services.printer_manager import PrinterInfo, printer_manager
  953. printer_manager._printer_info[55555] = PrinterInfo(name="N", serial_number="S")
  954. try:
  955. result = printer_state_to_dict(mock_state, printer_id=55555, model=None)
  956. assert "model" not in result
  957. assert result["name"] == "N"
  958. finally:
  959. printer_manager._printer_info.pop(55555, None)
  960. class TestStatusKeyDryingDedup:
  961. """Regression tests for WebSocket dedup including drying fields.
  962. The WebSocket broadcast deduplication uses printer_state_to_dict output
  963. to detect changes. If drying fields (like dry_time) are missing from
  964. the dict, changes to those fields won't trigger broadcasts.
  965. """
  966. def test_dry_time_change_changes_status_key(self):
  967. """Verify dry_time is present in AMS unit data so dedup can detect changes."""
  968. state = MagicMock()
  969. state.connected = True
  970. state.state = "IDLE"
  971. state.current_print = None
  972. state.subtask_name = None
  973. state.gcode_file = None
  974. state.progress = 0
  975. state.remaining_time = 0
  976. state.layer_num = 0
  977. state.total_layers = 0
  978. state.temperatures = {"nozzle": 25, "bed": 25}
  979. state.hms_errors = []
  980. state.ams_status_main = 0
  981. state.ams_status_sub = 0
  982. state.tray_now = None
  983. state.wifi_signal = -50
  984. state.stg_cur = -1
  985. # First state: drying active with 30 minutes remaining
  986. state.raw_data = {"ams": [{"id": 0, "dry_time": 30, "module_type": "n3f", "tray": [{"id": 0}]}]}
  987. result1 = printer_state_to_dict(state)
  988. # Second state: drying time decreased
  989. state.raw_data = {"ams": [{"id": 0, "dry_time": 29, "module_type": "n3f", "tray": [{"id": 0}]}]}
  990. result2 = printer_state_to_dict(state)
  991. # The dicts should differ — dry_time changed
  992. assert result1["ams"][0]["dry_time"] == 30
  993. assert result2["ams"][0]["dry_time"] == 29
  994. assert result1["ams"] != result2["ams"]
  995. class TestSupportsChamberTemp:
  996. """Tests for supports_chamber_temp helper function."""
  997. def test_x1_series_supported(self):
  998. """Verify X1 series printers support chamber temp."""
  999. assert supports_chamber_temp("X1") is True
  1000. assert supports_chamber_temp("X1C") is True
  1001. assert supports_chamber_temp("X1E") is True
  1002. def test_p2_series_supported(self):
  1003. """Verify P2 series printers support chamber temp."""
  1004. assert supports_chamber_temp("P2S") is True
  1005. def test_h2_series_supported(self):
  1006. """Verify H2 series printers support chamber temp."""
  1007. assert supports_chamber_temp("H2C") is True
  1008. assert supports_chamber_temp("H2D") is True
  1009. assert supports_chamber_temp("H2DPRO") is True
  1010. assert supports_chamber_temp("H2S") is True
  1011. def test_p1_series_not_supported(self):
  1012. """Verify P1 series printers do NOT support chamber temp."""
  1013. assert supports_chamber_temp("P1P") is False
  1014. assert supports_chamber_temp("P1S") is False
  1015. def test_a1_series_not_supported(self):
  1016. """Verify A1 series printers do NOT support chamber temp."""
  1017. assert supports_chamber_temp("A1") is False
  1018. assert supports_chamber_temp("A1MINI") is False
  1019. def test_none_model_not_supported(self):
  1020. """Verify None model returns False."""
  1021. assert supports_chamber_temp(None) is False
  1022. def test_case_insensitive(self):
  1023. """Verify model matching is case-insensitive."""
  1024. assert supports_chamber_temp("x1c") is True
  1025. assert supports_chamber_temp("X1c") is True
  1026. assert supports_chamber_temp("p1s") is False
  1027. def test_internal_model_codes_supported(self):
  1028. """Verify internal model codes from MQTT/SSDP are recognized."""
  1029. # X1/X1C
  1030. assert supports_chamber_temp("BL-P001") is True
  1031. # X1E
  1032. assert supports_chamber_temp("C13") is True
  1033. # H2D
  1034. assert supports_chamber_temp("O1D") is True
  1035. # H2C
  1036. assert supports_chamber_temp("O1C") is True
  1037. # H2S
  1038. assert supports_chamber_temp("O1S") is True
  1039. # H2D Pro
  1040. assert supports_chamber_temp("O1E") is True
  1041. # P2S
  1042. assert supports_chamber_temp("N7") is True
  1043. def test_internal_model_codes_not_supported(self):
  1044. """Verify A1/P1 internal codes are NOT supported."""
  1045. # P1P
  1046. assert supports_chamber_temp("C11") is False
  1047. # P1S
  1048. assert supports_chamber_temp("C12") is False
  1049. # A1
  1050. assert supports_chamber_temp("N2S") is False
  1051. # A1 Mini
  1052. assert supports_chamber_temp("N1") is False
  1053. class TestIsBedSlinger:
  1054. """Tests for is_bed_slinger helper function (#1334)."""
  1055. def test_a1_series_is_bed_slinger(self):
  1056. """A1 / A1 Mini are open-frame bed-slingers — Z axis is the toolhead."""
  1057. from backend.app.services.printer_manager import is_bed_slinger
  1058. assert is_bed_slinger("A1") is True
  1059. assert is_bed_slinger("A1 Mini") is True
  1060. assert is_bed_slinger("A1MINI") is True
  1061. assert is_bed_slinger("A1-MINI") is True
  1062. def test_a1_internal_codes_recognised(self):
  1063. """Internal MQTT/SSDP codes for A1 family must also classify as bed-slinger."""
  1064. from backend.app.services.printer_manager import is_bed_slinger
  1065. # A1 Mini
  1066. assert is_bed_slinger("N1") is True
  1067. # A1
  1068. assert is_bed_slinger("N2S") is True
  1069. def test_bed_on_z_models_not_bed_slingers(self):
  1070. """X1 / P1 / H2 / H2C / H2D / H2S / P2S all have the bed on Z."""
  1071. from backend.app.services.printer_manager import is_bed_slinger
  1072. for model in ("X1", "X1C", "X1E", "P1P", "P1S", "P2S", "H2C", "H2D", "H2DPRO", "H2S"):
  1073. assert is_bed_slinger(model) is False, f"{model} should NOT be classified as bed-slinger"
  1074. def test_none_model_returns_false(self):
  1075. from backend.app.services.printer_manager import is_bed_slinger
  1076. assert is_bed_slinger(None) is False
  1077. assert is_bed_slinger("") is False
  1078. def test_case_insensitive(self):
  1079. from backend.app.services.printer_manager import is_bed_slinger
  1080. assert is_bed_slinger("a1") is True
  1081. assert is_bed_slinger("a1 mini") is True
  1082. assert is_bed_slinger("x1c") is False
  1083. class TestSupportsDrying:
  1084. """Tests for supports_drying helper function."""
  1085. def test_known_supported_with_firmware(self):
  1086. """Verify known models with sufficient firmware return True."""
  1087. assert supports_drying("X1C", "01.09.00.00") is True
  1088. assert supports_drying("P1S", "01.08.00.00") is True
  1089. assert supports_drying("H2D", "01.02.30.00") is True
  1090. assert supports_drying("H2S", "01.02.00.00") is True
  1091. assert supports_drying("P2S", "01.02.00.00") is True
  1092. assert supports_drying("N7", "01.02.00.00") is True
  1093. def test_known_supported_old_firmware(self):
  1094. """Verify known models with old firmware return False."""
  1095. assert supports_drying("X1C", "01.08.00.00") is False
  1096. assert supports_drying("P1S", "01.07.00.00") is False
  1097. assert supports_drying("H2S", "01.01.00.00") is False
  1098. assert supports_drying("P2S", "01.01.99.99") is False
  1099. assert supports_drying("N7", "01.01.99.99") is False
  1100. def test_known_supported_no_firmware(self):
  1101. """Verify known models with no firmware return False."""
  1102. assert supports_drying("X1C", None) is False
  1103. assert supports_drying("P2S", None) is False
  1104. def test_unsupported_models(self):
  1105. """Verify models without AMS drying support return False regardless of firmware."""
  1106. for model in ["A1", "A1MINI", "A1-MINI", "H2C", "N1", "N2S"]:
  1107. assert supports_drying(model, "99.99.99.99") is False, f"Expected False for {model}"
  1108. def test_unknown_models_allowed(self):
  1109. """Verify unknown models are allowed (graceful fallback).
  1110. Models not in the unsupported set AND not matching any known firmware-gated
  1111. model substring get the benefit of the doubt and return True.
  1112. "H2D Pro" contains "H2D" so it IS firmware-gated (needs firmware).
  1113. """
  1114. # Truly unknown models: no substring match in _DRYING_MIN_FIRMWARE
  1115. assert supports_drying("FUTURE_MODEL", None) is True
  1116. # X1E contains "X1" substring, so it IS firmware-gated
  1117. assert supports_drying("X1E", "01.09.00.00") is True
  1118. # H2D Pro contains "H2D" substring, so it IS firmware-gated
  1119. assert supports_drying("H2D Pro", "01.02.30.00") is True
  1120. def test_none_model(self):
  1121. """Verify None model returns False."""
  1122. assert supports_drying(None, "01.09.00.00") is False
  1123. def test_case_insensitive(self):
  1124. """Verify model matching is case-insensitive."""
  1125. assert supports_drying("x1c", "01.09.00.00") is True
  1126. assert supports_drying("p2s", "01.02.00.00") is True
  1127. assert supports_drying("a1", "99.99.99.99") is False
  1128. class TestGetDerivedStatusName:
  1129. """Tests for get_derived_status_name function."""
  1130. def test_stg_cur_255_returns_none(self):
  1131. """Verify stg_cur=255 (A1/P1 idle) returns None, not 'Unknown stage (255)'."""
  1132. state = MagicMock()
  1133. state.stg_cur = 255
  1134. state.state = "IDLE"
  1135. result = get_derived_status_name(state)
  1136. assert result is None
  1137. def test_stg_cur_negative_one_returns_none_when_idle(self):
  1138. """Verify stg_cur=-1 (X1 idle) returns None."""
  1139. state = MagicMock()
  1140. state.stg_cur = -1
  1141. state.state = "IDLE"
  1142. result = get_derived_status_name(state)
  1143. assert result is None
  1144. def test_valid_stage_returns_name(self):
  1145. """Verify valid stg_cur values return stage name."""
  1146. state = MagicMock()
  1147. state.stg_cur = 1 # Auto bed leveling
  1148. result = get_derived_status_name(state)
  1149. assert result == "Auto bed leveling"
  1150. def test_stg_cur_zero_returns_printing(self):
  1151. """Verify stg_cur=0 returns 'Printing' when no model specified."""
  1152. state = MagicMock()
  1153. state.stg_cur = 0
  1154. result = get_derived_status_name(state)
  1155. assert result == "Printing"
  1156. def test_a1_idle_with_stg_cur_zero_returns_none(self):
  1157. """Verify A1 with IDLE state and stg_cur=0 returns None (bug workaround)."""
  1158. state = MagicMock()
  1159. state.stg_cur = 0
  1160. state.state = "IDLE"
  1161. # Test various A1 model names
  1162. for model in ["A1", "A1 Mini", "A1-Mini", "A1MINI", "N1", "N2S"]:
  1163. result = get_derived_status_name(state, model)
  1164. assert result is None, f"Expected None for model {model}"
  1165. def test_a1_running_with_stg_cur_zero_returns_printing(self):
  1166. """Verify A1 with RUNNING state and stg_cur=0 still returns 'Printing'."""
  1167. state = MagicMock()
  1168. state.stg_cur = 0
  1169. state.state = "RUNNING"
  1170. result = get_derived_status_name(state, "A1")
  1171. assert result == "Printing"
  1172. def test_non_a1_idle_with_stg_cur_zero_returns_printing(self):
  1173. """Verify non-A1 models with IDLE and stg_cur=0 still return 'Printing'."""
  1174. state = MagicMock()
  1175. state.stg_cur = 0
  1176. state.state = "IDLE"
  1177. # X1C should not get the workaround
  1178. result = get_derived_status_name(state, "X1C")
  1179. assert result == "Printing"
  1180. class TestHasStgCurIdleBug:
  1181. """Tests for has_stg_cur_idle_bug function."""
  1182. def test_a1_models_return_true(self):
  1183. """Verify A1 model variants return True."""
  1184. assert has_stg_cur_idle_bug("A1") is True
  1185. assert has_stg_cur_idle_bug("A1 Mini") is True
  1186. assert has_stg_cur_idle_bug("A1-Mini") is True
  1187. assert has_stg_cur_idle_bug("A1MINI") is True
  1188. assert has_stg_cur_idle_bug("a1") is True # case insensitive
  1189. assert has_stg_cur_idle_bug("a1 mini") is True
  1190. def test_p1_models_return_true(self):
  1191. """Verify P1P/P1S model variants return True."""
  1192. assert has_stg_cur_idle_bug("P1P") is True
  1193. assert has_stg_cur_idle_bug("P1S") is True
  1194. assert has_stg_cur_idle_bug("p1p") is True # case insensitive
  1195. def test_internal_codes_return_true(self):
  1196. """Verify internal model codes return True."""
  1197. assert has_stg_cur_idle_bug("N1") is True # A1 Mini
  1198. assert has_stg_cur_idle_bug("N2S") is True # A1
  1199. assert has_stg_cur_idle_bug("C11") is True # P1P
  1200. assert has_stg_cur_idle_bug("C12") is True # P1S
  1201. def test_non_affected_models_return_false(self):
  1202. """Verify non-affected models return False."""
  1203. assert has_stg_cur_idle_bug("X1C") is False
  1204. assert has_stg_cur_idle_bug("X1") is False
  1205. assert has_stg_cur_idle_bug("H2D") is False
  1206. def test_none_model_returns_false(self):
  1207. """Verify None model returns False."""
  1208. assert has_stg_cur_idle_bug(None) is False
  1209. def test_empty_model_returns_false(self):
  1210. """Verify empty model returns False."""
  1211. assert has_stg_cur_idle_bug("") is False
  1212. class TestInitPrinterConnections:
  1213. """Tests for init_printer_connections function."""
  1214. @pytest.mark.asyncio
  1215. async def test_connects_all_active_printers(self):
  1216. """Verify all active printers are connected."""
  1217. mock_db = AsyncMock()
  1218. mock_printer1 = MagicMock(id=1, is_active=True)
  1219. mock_printer2 = MagicMock(id=2, is_active=True)
  1220. mock_result = MagicMock()
  1221. mock_result.scalars.return_value.all.return_value = [mock_printer1, mock_printer2]
  1222. mock_db.execute.return_value = mock_result
  1223. with patch("backend.app.services.printer_manager.printer_manager") as mock_manager:
  1224. mock_manager.connect_printer = AsyncMock()
  1225. await init_printer_connections(mock_db)
  1226. assert mock_manager.connect_printer.call_count == 2
  1227. @pytest.mark.asyncio
  1228. async def test_handles_empty_printer_list(self):
  1229. """Verify empty printer list is handled."""
  1230. mock_db = AsyncMock()
  1231. mock_result = MagicMock()
  1232. mock_result.scalars.return_value.all.return_value = []
  1233. mock_db.execute.return_value = mock_result
  1234. with patch("backend.app.services.printer_manager.printer_manager") as mock_manager:
  1235. mock_manager.connect_printer = AsyncMock()
  1236. await init_printer_connections(mock_db)
  1237. mock_manager.connect_printer.assert_not_called()
  1238. class TestAmsChangeCallback:
  1239. """Tests for AMS change callback functionality."""
  1240. @pytest.fixture
  1241. def manager(self):
  1242. """Create a fresh PrinterManager instance."""
  1243. return PrinterManager()
  1244. def test_ams_change_callback_is_triggered(self, manager):
  1245. """Verify AMS change callback is called when AMS data changes."""
  1246. callback = MagicMock()
  1247. manager.set_ams_change_callback(callback)
  1248. # Verify callback was set
  1249. assert manager._on_ams_change == callback
  1250. def test_ams_change_callback_receives_correct_data(self, manager):
  1251. """Verify AMS change callback receives the correct AMS data format."""
  1252. received_data = []
  1253. def capture_callback(printer_id, ams_data):
  1254. received_data.append((printer_id, ams_data))
  1255. manager.set_ams_change_callback(capture_callback)
  1256. # The callback should accept printer_id and ams_data
  1257. # This tests the callback signature
  1258. assert manager._on_ams_change is not None
  1259. assert callable(manager._on_ams_change)
  1260. class TestParsePlateId:
  1261. """Tests for parse_plate_id() — active-print plate extraction from gcode paths.
  1262. Regression coverage for #881 follow-up: the REST /status endpoint and the
  1263. WebSocket push path both use this helper, so they must agree on the plate
  1264. number the frontend sees.
  1265. """
  1266. def test_bambu_metadata_path(self):
  1267. # Canonical path that Bambu Studio / OrcaSlicer stamp into the 3MF.
  1268. assert parse_plate_id("/Metadata/plate_2.gcode") == 2
  1269. def test_plate_one(self):
  1270. assert parse_plate_id("/Metadata/plate_1.gcode") == 1
  1271. def test_double_digit_plate(self):
  1272. assert parse_plate_id("/Metadata/plate_12.gcode") == 12
  1273. def test_none_input(self):
  1274. assert parse_plate_id(None) is None
  1275. def test_empty_string(self):
  1276. assert parse_plate_id("") is None
  1277. def test_path_without_plate_segment(self):
  1278. # Some firmware / slicers report a bare filename without the plate marker.
  1279. assert parse_plate_id("/upload/my-model.gcode") is None
  1280. def test_similar_but_non_matching_names(self):
  1281. # "plate.gcode" (no number) and "nameplate_2.gcode" (substring) must not
  1282. # be mistaken for real plate markers. The regex anchors on `plate_<num>`.
  1283. assert parse_plate_id("/Metadata/plate.gcode") is None
  1284. assert parse_plate_id("/plates/3.gcode") is None
  1285. def test_substring_match_still_extracts(self):
  1286. # The regex isn't anchored to the start of a segment — any occurrence
  1287. # wins. This matches real Bambu paths where the segment is preceded by
  1288. # arbitrary directory noise, and matches the equivalent frontend regex.
  1289. assert parse_plate_id("/uploads/project/plate_5.gcode.md5") == 5
  1290. class TestResolvePlateId:
  1291. """Tests for resolve_plate_id() — plate resolution with dispatch precedence.
  1292. Regression coverage for #1166: P1S firmware 01.10.00.00 only puts the .3mf
  1293. filename in print.gcode_file, so parse_plate_id() returns None and the
  1294. printer card falls back to plate 1. When Bambuddy dispatches the print
  1295. itself we know the right plate; resolve_plate_id() prefers that record over
  1296. the gcode_file regex when subtask_name matches.
  1297. """
  1298. def _make_state(self, **kwargs):
  1299. from backend.app.services.bambu_mqtt import PrinterState
  1300. state = PrinterState()
  1301. for k, v in kwargs.items():
  1302. setattr(state, k, v)
  1303. return state
  1304. def test_dispatched_plate_wins_when_subtask_matches(self):
  1305. # User dispatches plate 4 via Bambuddy. Printer reflects subtask_name
  1306. # but firmware drops the plate path from gcode_file. Without the dispatch
  1307. # record we'd default to plate 1.
  1308. from backend.app.services.printer_manager import resolve_plate_id
  1309. state = self._make_state(
  1310. gcode_file="MyModel.3mf", # No plate path — firmware bug
  1311. subtask_name="MyModel",
  1312. dispatched_plate_id=4,
  1313. dispatched_subtask="MyModel",
  1314. )
  1315. assert resolve_plate_id(state) == 4
  1316. def test_dispatched_ignored_when_subtask_differs(self):
  1317. # Bambuddy's dispatch record is for a previous print; the printer is
  1318. # now running a different subtask (Studio-direct dispatch). The stale
  1319. # record must not be used — fall back to gcode_file regex.
  1320. from backend.app.services.printer_manager import resolve_plate_id
  1321. state = self._make_state(
  1322. gcode_file="/Metadata/plate_2.gcode",
  1323. subtask_name="DifferentPrint",
  1324. dispatched_plate_id=4,
  1325. dispatched_subtask="MyModel",
  1326. )
  1327. assert resolve_plate_id(state) == 2
  1328. def test_falls_back_to_gcode_regex_without_dispatch(self):
  1329. # Studio-direct dispatch — no Bambuddy dispatch record. Existing logic
  1330. # (parse_plate_id on gcode_file) must still work.
  1331. from backend.app.services.printer_manager import resolve_plate_id
  1332. state = self._make_state(
  1333. gcode_file="/Metadata/plate_3.gcode",
  1334. subtask_name="MyModel",
  1335. )
  1336. assert resolve_plate_id(state) == 3
  1337. def test_returns_none_when_nothing_resolvable(self):
  1338. # No dispatch record AND firmware swallowed the plate path. The route
  1339. # uses this signal to invoke the 3MF-scan fallback.
  1340. from backend.app.services.printer_manager import resolve_plate_id
  1341. state = self._make_state(
  1342. gcode_file="MyModel.3mf",
  1343. subtask_name="MyModel",
  1344. )
  1345. assert resolve_plate_id(state) is None
  1346. def test_dispatched_subtask_required_to_avoid_false_match(self):
  1347. # dispatched_plate_id without dispatched_subtask is incomplete — we
  1348. # can't validate it points at the current print, so we ignore it.
  1349. from backend.app.services.printer_manager import resolve_plate_id
  1350. state = self._make_state(
  1351. gcode_file="MyModel.3mf",
  1352. subtask_name="MyModel",
  1353. dispatched_plate_id=4,
  1354. dispatched_subtask=None,
  1355. )
  1356. assert resolve_plate_id(state) is None