test_printer_manager.py 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778
  1. """Unit tests for PrinterManager service.
  2. Tests printer connection management, status tracking, and print control.
  3. """
  4. import asyncio
  5. import pytest
  6. from unittest.mock import MagicMock, AsyncMock, patch, PropertyMock
  7. from datetime import datetime
  8. from backend.app.services.printer_manager import (
  9. PrinterManager,
  10. printer_state_to_dict,
  11. init_printer_connections,
  12. )
  13. class TestPrinterManager:
  14. """Tests for PrinterManager class."""
  15. @pytest.fixture
  16. def manager(self):
  17. """Create a fresh PrinterManager instance."""
  18. return PrinterManager()
  19. @pytest.fixture
  20. def mock_printer(self):
  21. """Create a mock Printer object."""
  22. printer = MagicMock()
  23. printer.id = 1
  24. printer.ip_address = "192.168.1.100"
  25. printer.serial_number = "00M09A123456789"
  26. printer.access_code = "12345678"
  27. printer.is_active = True
  28. return printer
  29. @pytest.fixture
  30. def mock_client(self):
  31. """Create a mock BambuMQTTClient."""
  32. client = MagicMock()
  33. client.state = MagicMock()
  34. client.state.connected = True
  35. client.state.state = "IDLE"
  36. client.state.progress = 0
  37. client.state.temperatures = {"nozzle": 25, "bed": 25}
  38. client.state.raw_data = {}
  39. client.logging_enabled = False
  40. return client
  41. # ========================================================================
  42. # Tests for initialization
  43. # ========================================================================
  44. def test_init_creates_empty_clients_dict(self, manager):
  45. """Verify manager initializes with empty clients dict."""
  46. assert manager._clients == {}
  47. def test_init_callbacks_are_none(self, manager):
  48. """Verify all callbacks are initially None."""
  49. assert manager._on_print_start is None
  50. assert manager._on_print_complete is None
  51. assert manager._on_status_change is None
  52. assert manager._on_ams_change is None
  53. def test_init_loop_is_none(self, manager):
  54. """Verify event loop is initially None."""
  55. assert manager._loop is None
  56. # ========================================================================
  57. # Tests for callback setters
  58. # ========================================================================
  59. def test_set_event_loop(self, manager):
  60. """Verify event loop can be set."""
  61. mock_loop = MagicMock()
  62. manager.set_event_loop(mock_loop)
  63. assert manager._loop == mock_loop
  64. def test_set_print_start_callback(self, manager):
  65. """Verify print start callback can be set."""
  66. callback = MagicMock()
  67. manager.set_print_start_callback(callback)
  68. assert manager._on_print_start == callback
  69. def test_set_print_complete_callback(self, manager):
  70. """Verify print complete callback can be set."""
  71. callback = MagicMock()
  72. manager.set_print_complete_callback(callback)
  73. assert manager._on_print_complete == callback
  74. def test_set_status_change_callback(self, manager):
  75. """Verify status change callback can be set."""
  76. callback = MagicMock()
  77. manager.set_status_change_callback(callback)
  78. assert manager._on_status_change == callback
  79. def test_set_ams_change_callback(self, manager):
  80. """Verify AMS change callback can be set."""
  81. callback = MagicMock()
  82. manager.set_ams_change_callback(callback)
  83. assert manager._on_ams_change == callback
  84. # ========================================================================
  85. # Tests for _schedule_async
  86. # ========================================================================
  87. def test_schedule_async_with_running_loop(self, manager):
  88. """Verify async coroutine is scheduled when loop is running."""
  89. mock_loop = MagicMock()
  90. mock_loop.is_running.return_value = True
  91. manager._loop = mock_loop
  92. async def dummy_coro():
  93. pass
  94. coro = dummy_coro()
  95. manager._schedule_async(coro)
  96. mock_loop.is_running.assert_called_once()
  97. # Clean up the coroutine
  98. coro.close()
  99. def test_schedule_async_without_loop(self, manager):
  100. """Verify nothing happens when no loop is set."""
  101. async def dummy_coro():
  102. pass
  103. coro = dummy_coro()
  104. # Should not raise
  105. manager._schedule_async(coro)
  106. coro.close()
  107. def test_schedule_async_with_stopped_loop(self, manager):
  108. """Verify nothing happens when loop is not running."""
  109. mock_loop = MagicMock()
  110. mock_loop.is_running.return_value = False
  111. manager._loop = mock_loop
  112. async def dummy_coro():
  113. pass
  114. coro = dummy_coro()
  115. manager._schedule_async(coro)
  116. coro.close()
  117. # ========================================================================
  118. # Tests for connect_printer
  119. # ========================================================================
  120. @pytest.mark.asyncio
  121. async def test_connect_printer_creates_client(self, manager, mock_printer):
  122. """Verify connecting creates an MQTT client."""
  123. with patch(
  124. 'backend.app.services.printer_manager.BambuMQTTClient'
  125. ) as MockClient:
  126. mock_instance = MagicMock()
  127. mock_instance.state = MagicMock()
  128. mock_instance.state.connected = True
  129. MockClient.return_value = mock_instance
  130. result = await manager.connect_printer(mock_printer)
  131. MockClient.assert_called_once()
  132. mock_instance.connect.assert_called_once()
  133. assert mock_printer.id in manager._clients
  134. assert result is True
  135. @pytest.mark.asyncio
  136. async def test_connect_printer_disconnects_existing(self, manager, mock_printer, mock_client):
  137. """Verify connecting disconnects existing client first."""
  138. manager._clients[mock_printer.id] = mock_client
  139. with patch(
  140. 'backend.app.services.printer_manager.BambuMQTTClient'
  141. ) 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(
  152. 'backend.app.services.printer_manager.BambuMQTTClient'
  153. ) as MockClient:
  154. mock_instance = MagicMock()
  155. mock_instance.state = MagicMock()
  156. mock_instance.state.connected = False
  157. MockClient.return_value = mock_instance
  158. result = await manager.connect_printer(mock_printer)
  159. assert result is False
  160. # ========================================================================
  161. # Tests for disconnect_printer
  162. # ========================================================================
  163. def test_disconnect_printer_removes_client(self, manager, mock_client):
  164. """Verify disconnecting removes and disconnects client."""
  165. manager._clients[1] = mock_client
  166. manager.disconnect_printer(1)
  167. mock_client.disconnect.assert_called_once()
  168. assert 1 not in manager._clients
  169. def test_disconnect_printer_handles_missing(self, manager):
  170. """Verify disconnecting non-existent printer doesn't raise."""
  171. manager.disconnect_printer(999) # Should not raise
  172. # ========================================================================
  173. # Tests for disconnect_all
  174. # ========================================================================
  175. def test_disconnect_all_disconnects_all_clients(self, manager):
  176. """Verify all clients are disconnected."""
  177. client1 = MagicMock()
  178. client2 = MagicMock()
  179. manager._clients[1] = client1
  180. manager._clients[2] = client2
  181. manager.disconnect_all()
  182. client1.disconnect.assert_called_once()
  183. client2.disconnect.assert_called_once()
  184. assert len(manager._clients) == 0
  185. # ========================================================================
  186. # Tests for get_status
  187. # ========================================================================
  188. def test_get_status_returns_state(self, manager, mock_client):
  189. """Verify get_status returns client state."""
  190. manager._clients[1] = mock_client
  191. result = manager.get_status(1)
  192. mock_client.check_staleness.assert_called_once()
  193. assert result == mock_client.state
  194. def test_get_status_returns_none_for_unknown(self, manager):
  195. """Verify get_status returns None for unknown printer."""
  196. result = manager.get_status(999)
  197. assert result is None
  198. # ========================================================================
  199. # Tests for get_all_statuses
  200. # ========================================================================
  201. def test_get_all_statuses_returns_all(self, manager):
  202. """Verify all statuses are returned."""
  203. client1 = MagicMock()
  204. client1.state = MagicMock(connected=True)
  205. client2 = MagicMock()
  206. client2.state = MagicMock(connected=False)
  207. manager._clients[1] = client1
  208. manager._clients[2] = client2
  209. result = manager.get_all_statuses()
  210. assert len(result) == 2
  211. assert 1 in result
  212. assert 2 in result
  213. client1.check_staleness.assert_called_once()
  214. client2.check_staleness.assert_called_once()
  215. # ========================================================================
  216. # Tests for is_connected
  217. # ========================================================================
  218. def test_is_connected_returns_true(self, manager, mock_client):
  219. """Verify is_connected returns True for connected printer."""
  220. mock_client.check_staleness.return_value = True
  221. manager._clients[1] = mock_client
  222. result = manager.is_connected(1)
  223. assert result is True
  224. def test_is_connected_returns_false_for_unknown(self, manager):
  225. """Verify is_connected returns False for unknown printer."""
  226. result = manager.is_connected(999)
  227. assert result is False
  228. # ========================================================================
  229. # Tests for get_client
  230. # ========================================================================
  231. def test_get_client_returns_client(self, manager, mock_client):
  232. """Verify get_client returns the client."""
  233. manager._clients[1] = mock_client
  234. result = manager.get_client(1)
  235. assert result == mock_client
  236. def test_get_client_returns_none_for_unknown(self, manager):
  237. """Verify get_client returns None for unknown printer."""
  238. result = manager.get_client(999)
  239. assert result is None
  240. # ========================================================================
  241. # Tests for mark_printer_offline
  242. # ========================================================================
  243. def test_mark_printer_offline_updates_state(self, manager, mock_client):
  244. """Verify mark_printer_offline updates client state."""
  245. mock_client.state.connected = True
  246. manager._clients[1] = mock_client
  247. manager.mark_printer_offline(1)
  248. assert mock_client.state.connected is False
  249. assert mock_client.state.state == "unknown"
  250. def test_mark_printer_offline_triggers_callback(self, manager, mock_client):
  251. """Verify mark_printer_offline triggers status callback."""
  252. mock_client.state.connected = True
  253. manager._clients[1] = mock_client
  254. # Callback must return a coroutine
  255. async def async_callback(printer_id, state):
  256. pass
  257. manager._on_status_change = async_callback
  258. # Need a running loop for callback
  259. mock_loop = MagicMock()
  260. mock_loop.is_running.return_value = True
  261. manager._loop = mock_loop
  262. manager.mark_printer_offline(1)
  263. # Callback should be scheduled via run_coroutine_threadsafe
  264. mock_loop.is_running.assert_called()
  265. # State should be updated
  266. assert mock_client.state.connected is False
  267. def test_mark_printer_offline_handles_unknown(self, manager):
  268. """Verify mark_printer_offline handles unknown printer."""
  269. manager.mark_printer_offline(999) # Should not raise
  270. def test_mark_printer_offline_skips_already_offline(self, manager, mock_client):
  271. """Verify mark_printer_offline skips already offline printer."""
  272. mock_client.state.connected = False
  273. manager._clients[1] = mock_client
  274. manager.mark_printer_offline(1)
  275. # State should remain unchanged
  276. assert mock_client.state.connected is False
  277. # ========================================================================
  278. # Tests for start_print
  279. # ========================================================================
  280. def test_start_print_calls_client(self, manager, mock_client):
  281. """Verify start_print calls client method."""
  282. mock_client.start_print.return_value = True
  283. manager._clients[1] = mock_client
  284. result = manager.start_print(1, "test.gcode")
  285. mock_client.start_print.assert_called_once_with("test.gcode")
  286. assert result is True
  287. def test_start_print_returns_false_for_unknown(self, manager):
  288. """Verify start_print returns False for unknown printer."""
  289. result = manager.start_print(999, "test.gcode")
  290. assert result is False
  291. # ========================================================================
  292. # Tests for stop_print
  293. # ========================================================================
  294. def test_stop_print_calls_client(self, manager, mock_client):
  295. """Verify stop_print calls client method."""
  296. mock_client.stop_print.return_value = True
  297. manager._clients[1] = mock_client
  298. result = manager.stop_print(1)
  299. mock_client.stop_print.assert_called_once()
  300. assert result is True
  301. def test_stop_print_returns_false_for_unknown(self, manager):
  302. """Verify stop_print returns False for unknown printer."""
  303. result = manager.stop_print(999)
  304. assert result is False
  305. # ========================================================================
  306. # Tests for wait_for_cooldown
  307. # ========================================================================
  308. @pytest.mark.asyncio
  309. async def test_wait_for_cooldown_returns_true_when_cool(self, manager, mock_client):
  310. """Verify wait_for_cooldown returns True when printer is cool."""
  311. mock_client.state.connected = True
  312. mock_client.state.temperatures = {"nozzle": 40, "bed": 30}
  313. mock_client.check_staleness.return_value = True
  314. manager._clients[1] = mock_client
  315. result = await manager.wait_for_cooldown(1, target_temp=50)
  316. assert result is True
  317. @pytest.mark.asyncio
  318. async def test_wait_for_cooldown_returns_false_on_disconnect(self, manager, mock_client):
  319. """Verify wait_for_cooldown returns False when printer disconnects."""
  320. mock_client.state.connected = False
  321. mock_client.check_staleness.return_value = False
  322. manager._clients[1] = mock_client
  323. result = await manager.wait_for_cooldown(1, target_temp=50, timeout=1)
  324. assert result is False
  325. @pytest.mark.asyncio
  326. async def test_wait_for_cooldown_returns_false_for_unknown(self, manager):
  327. """Verify wait_for_cooldown returns False for unknown printer."""
  328. result = await manager.wait_for_cooldown(999, target_temp=50, timeout=1)
  329. assert result is False
  330. @pytest.mark.asyncio
  331. async def test_wait_for_cooldown_checks_both_nozzles(self, manager, mock_client):
  332. """Verify wait_for_cooldown checks both nozzles for dual extruders."""
  333. mock_client.state.connected = True
  334. mock_client.state.temperatures = {"nozzle": 40, "nozzle_2": 45, "bed": 30}
  335. mock_client.check_staleness.return_value = True
  336. manager._clients[1] = mock_client
  337. result = await manager.wait_for_cooldown(1, target_temp=50)
  338. assert result is True
  339. # ========================================================================
  340. # Tests for logging methods
  341. # ========================================================================
  342. def test_enable_logging_calls_client(self, manager, mock_client):
  343. """Verify enable_logging calls client method."""
  344. manager._clients[1] = mock_client
  345. result = manager.enable_logging(1, True)
  346. mock_client.enable_logging.assert_called_once_with(True)
  347. assert result is True
  348. def test_enable_logging_returns_false_for_unknown(self, manager):
  349. """Verify enable_logging returns False for unknown printer."""
  350. result = manager.enable_logging(999, True)
  351. assert result is False
  352. def test_get_logs_returns_logs(self, manager, mock_client):
  353. """Verify get_logs returns client logs."""
  354. mock_logs = [MagicMock(), MagicMock()]
  355. mock_client.get_logs.return_value = mock_logs
  356. manager._clients[1] = mock_client
  357. result = manager.get_logs(1)
  358. assert result == mock_logs
  359. def test_get_logs_returns_empty_for_unknown(self, manager):
  360. """Verify get_logs returns empty list for unknown printer."""
  361. result = manager.get_logs(999)
  362. assert result == []
  363. def test_clear_logs_calls_client(self, manager, mock_client):
  364. """Verify clear_logs calls client method."""
  365. manager._clients[1] = mock_client
  366. result = manager.clear_logs(1)
  367. mock_client.clear_logs.assert_called_once()
  368. assert result is True
  369. def test_clear_logs_returns_false_for_unknown(self, manager):
  370. """Verify clear_logs returns False for unknown printer."""
  371. result = manager.clear_logs(999)
  372. assert result is False
  373. def test_is_logging_enabled_returns_status(self, manager, mock_client):
  374. """Verify is_logging_enabled returns client status."""
  375. mock_client.logging_enabled = True
  376. manager._clients[1] = mock_client
  377. result = manager.is_logging_enabled(1)
  378. assert result is True
  379. def test_is_logging_enabled_returns_false_for_unknown(self, manager):
  380. """Verify is_logging_enabled returns False for unknown printer."""
  381. result = manager.is_logging_enabled(999)
  382. assert result is False
  383. # ========================================================================
  384. # Tests for request_status_update
  385. # ========================================================================
  386. def test_request_status_update_calls_client(self, manager, mock_client):
  387. """Verify request_status_update calls client method."""
  388. mock_client.request_status_update.return_value = True
  389. manager._clients[1] = mock_client
  390. result = manager.request_status_update(1)
  391. mock_client.request_status_update.assert_called_once()
  392. assert result is True
  393. def test_request_status_update_returns_false_for_unknown(self, manager):
  394. """Verify request_status_update returns False for unknown printer."""
  395. result = manager.request_status_update(999)
  396. assert result is False
  397. # ========================================================================
  398. # Tests for test_connection
  399. # ========================================================================
  400. @pytest.mark.asyncio
  401. async def test_test_connection_success(self, manager):
  402. """Verify test_connection returns success on connection."""
  403. with patch(
  404. 'backend.app.services.printer_manager.BambuMQTTClient'
  405. ) as MockClient:
  406. mock_instance = MagicMock()
  407. mock_instance.state = MagicMock()
  408. mock_instance.state.connected = True
  409. mock_instance.state.state = "IDLE"
  410. mock_instance.state.raw_data = {"device_model": "X1C"}
  411. MockClient.return_value = mock_instance
  412. result = await manager.test_connection(
  413. "192.168.1.100", "00M09A123456789", "12345678"
  414. )
  415. assert result["success"] is True
  416. assert result["state"] == "IDLE"
  417. assert result["model"] == "X1C"
  418. mock_instance.disconnect.assert_called_once()
  419. @pytest.mark.asyncio
  420. async def test_test_connection_failure(self, manager):
  421. """Verify test_connection returns failure on connection error."""
  422. with patch(
  423. 'backend.app.services.printer_manager.BambuMQTTClient'
  424. ) as MockClient:
  425. mock_instance = MagicMock()
  426. mock_instance.state = MagicMock()
  427. mock_instance.state.connected = False
  428. MockClient.return_value = mock_instance
  429. result = await manager.test_connection(
  430. "192.168.1.100", "00M09A123456789", "12345678"
  431. )
  432. assert result["success"] is False
  433. assert result["state"] is None
  434. mock_instance.disconnect.assert_called_once()
  435. class TestPrinterStateToDict:
  436. """Tests for printer_state_to_dict helper function."""
  437. @pytest.fixture
  438. def mock_state(self):
  439. """Create a mock PrinterState."""
  440. state = MagicMock()
  441. state.connected = True
  442. state.state = "RUNNING"
  443. state.current_print = "test.3mf"
  444. state.subtask_name = "Test Print"
  445. state.gcode_file = "/sdcard/test.gcode"
  446. state.progress = 50
  447. state.remaining_time = 3600
  448. state.layer_num = 10
  449. state.total_layers = 20
  450. state.temperatures = {"nozzle": 200, "bed": 60}
  451. state.hms_errors = []
  452. state.ams_status_main = 0
  453. state.ams_status_sub = 0
  454. state.tray_now = "1"
  455. state.wifi_signal = -50
  456. state.raw_data = {}
  457. return state
  458. def test_basic_conversion(self, mock_state):
  459. """Verify basic state fields are converted."""
  460. result = printer_state_to_dict(mock_state)
  461. assert result["connected"] is True
  462. assert result["state"] == "RUNNING"
  463. assert result["progress"] == 50
  464. assert result["temperatures"] == {"nozzle": 200, "bed": 60}
  465. def test_ams_data_parsing(self, mock_state):
  466. """Verify AMS data is parsed correctly."""
  467. mock_state.raw_data = {
  468. "ams": [{
  469. "id": 0,
  470. "humidity_raw": 45,
  471. "temp": 25,
  472. "tray": [
  473. {
  474. "id": 0,
  475. "tray_color": "FF0000",
  476. "tray_type": "PLA",
  477. "tray_sub_brands": "Generic",
  478. "remain": 80,
  479. "k": 0.5,
  480. "tag_uid": "ABC123",
  481. "tray_uuid": "uuid-123",
  482. }
  483. ]
  484. }]
  485. }
  486. result = printer_state_to_dict(mock_state)
  487. assert result["ams"] is not None
  488. assert len(result["ams"]) == 1
  489. assert result["ams"][0]["humidity"] == 45
  490. assert len(result["ams"][0]["tray"]) == 1
  491. assert result["ams"][0]["tray"][0]["tray_color"] == "FF0000"
  492. def test_empty_tag_uid_becomes_none(self, mock_state):
  493. """Verify empty tag_uid is converted to None."""
  494. mock_state.raw_data = {
  495. "ams": [{
  496. "id": 0,
  497. "tray": [{
  498. "id": 0,
  499. "tag_uid": "",
  500. "tray_uuid": "00000000000000000000000000000000",
  501. }]
  502. }]
  503. }
  504. result = printer_state_to_dict(mock_state)
  505. assert result["ams"][0]["tray"][0]["tag_uid"] is None
  506. assert result["ams"][0]["tray"][0]["tray_uuid"] is None
  507. def test_zero_tag_uid_becomes_none(self, mock_state):
  508. """Verify zero tag_uid is converted to None."""
  509. mock_state.raw_data = {
  510. "ams": [{
  511. "id": 0,
  512. "tray": [{
  513. "id": 0,
  514. "tag_uid": "0000000000000000",
  515. }]
  516. }]
  517. }
  518. result = printer_state_to_dict(mock_state)
  519. assert result["ams"][0]["tray"][0]["tag_uid"] is None
  520. def test_vt_tray_parsing(self, mock_state):
  521. """Verify virtual tray is parsed correctly."""
  522. mock_state.raw_data = {
  523. "vt_tray": {
  524. "tray_color": "00FF00",
  525. "tray_type": "PETG",
  526. "tray_sub_brands": "Generic",
  527. "remain": 60,
  528. "tag_uid": "VT123",
  529. }
  530. }
  531. result = printer_state_to_dict(mock_state)
  532. assert result["vt_tray"] is not None
  533. assert result["vt_tray"]["id"] == 254
  534. assert result["vt_tray"]["tray_color"] == "00FF00"
  535. assert result["vt_tray"]["tray_type"] == "PETG"
  536. def test_hms_errors_conversion(self, mock_state):
  537. """Verify HMS errors are converted correctly."""
  538. error = MagicMock()
  539. error.code = "0700_0100"
  540. error.attr = 1
  541. error.module = "AMS"
  542. error.severity = 2
  543. mock_state.hms_errors = [error]
  544. result = printer_state_to_dict(mock_state)
  545. assert len(result["hms_errors"]) == 1
  546. assert result["hms_errors"][0]["code"] == "0700_0100"
  547. assert result["hms_errors"][0]["module"] == "AMS"
  548. def test_cover_url_added_for_running_print(self, mock_state):
  549. """Verify cover_url is added for running prints."""
  550. result = printer_state_to_dict(mock_state, printer_id=1)
  551. assert result["cover_url"] == "/api/v1/printers/1/cover"
  552. def test_cover_url_none_when_not_running(self, mock_state):
  553. """Verify cover_url is None when not printing."""
  554. mock_state.state = "IDLE"
  555. result = printer_state_to_dict(mock_state, printer_id=1)
  556. assert result["cover_url"] is None
  557. def test_ams_ht_detection(self, mock_state):
  558. """Verify AMS-HT is detected (1 tray vs 4)."""
  559. mock_state.raw_data = {
  560. "ams": [{
  561. "id": 0,
  562. "tray": [{"id": 0}] # Only 1 tray = AMS-HT
  563. }]
  564. }
  565. result = printer_state_to_dict(mock_state)
  566. assert result["ams"][0]["is_ams_ht"] is True
  567. def test_regular_ams_detection(self, mock_state):
  568. """Verify regular AMS is detected (4 trays)."""
  569. mock_state.raw_data = {
  570. "ams": [{
  571. "id": 0,
  572. "tray": [{"id": 0}, {"id": 1}, {"id": 2}, {"id": 3}]
  573. }]
  574. }
  575. result = printer_state_to_dict(mock_state)
  576. assert result["ams"][0]["is_ams_ht"] is False
  577. class TestInitPrinterConnections:
  578. """Tests for init_printer_connections function."""
  579. @pytest.mark.asyncio
  580. async def test_connects_all_active_printers(self):
  581. """Verify all active printers are connected."""
  582. mock_db = AsyncMock()
  583. mock_printer1 = MagicMock(id=1, is_active=True)
  584. mock_printer2 = MagicMock(id=2, is_active=True)
  585. mock_result = MagicMock()
  586. mock_result.scalars.return_value.all.return_value = [mock_printer1, mock_printer2]
  587. mock_db.execute.return_value = mock_result
  588. with patch(
  589. 'backend.app.services.printer_manager.printer_manager'
  590. ) as mock_manager:
  591. mock_manager.connect_printer = AsyncMock()
  592. await init_printer_connections(mock_db)
  593. assert mock_manager.connect_printer.call_count == 2
  594. @pytest.mark.asyncio
  595. async def test_handles_empty_printer_list(self):
  596. """Verify empty printer list is handled."""
  597. mock_db = AsyncMock()
  598. mock_result = MagicMock()
  599. mock_result.scalars.return_value.all.return_value = []
  600. mock_db.execute.return_value = mock_result
  601. with patch(
  602. 'backend.app.services.printer_manager.printer_manager'
  603. ) as mock_manager:
  604. mock_manager.connect_printer = AsyncMock()
  605. await init_printer_connections(mock_db)
  606. mock_manager.connect_printer.assert_not_called()