test_printer_manager.py 28 KB

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