test_printers_api.py 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951
  1. """Integration tests for Printers API endpoints.
  2. Tests the full request/response cycle for /api/v1/printers/ endpoints.
  3. """
  4. from unittest.mock import MagicMock, patch
  5. import pytest
  6. from httpx import AsyncClient
  7. class TestPrintersAPI:
  8. """Integration tests for /api/v1/printers/ endpoints."""
  9. # ========================================================================
  10. # List endpoints
  11. # ========================================================================
  12. @pytest.mark.asyncio
  13. @pytest.mark.integration
  14. async def test_list_printers_empty(self, async_client: AsyncClient):
  15. """Verify empty list is returned when no printers exist."""
  16. response = await async_client.get("/api/v1/printers/")
  17. assert response.status_code == 200
  18. assert response.json() == []
  19. @pytest.mark.asyncio
  20. @pytest.mark.integration
  21. async def test_list_printers_with_data(self, async_client: AsyncClient, printer_factory, db_session):
  22. """Verify list returns existing printers."""
  23. await printer_factory(name="Test Printer")
  24. response = await async_client.get("/api/v1/printers/")
  25. assert response.status_code == 200
  26. data = response.json()
  27. assert len(data) >= 1
  28. assert any(p["name"] == "Test Printer" for p in data)
  29. # ========================================================================
  30. # Create endpoints
  31. # ========================================================================
  32. @pytest.mark.asyncio
  33. @pytest.mark.integration
  34. async def test_create_printer(self, async_client: AsyncClient):
  35. """Verify printer can be created."""
  36. data = {
  37. "name": "New Printer",
  38. "serial_number": "00M09A111111111",
  39. "ip_address": "192.168.1.100",
  40. "access_code": "12345678",
  41. "is_active": True,
  42. "model": "X1C",
  43. }
  44. response = await async_client.post("/api/v1/printers/", json=data)
  45. assert response.status_code == 200
  46. result = response.json()
  47. assert result["name"] == "New Printer"
  48. assert result["serial_number"] == "00M09A111111111"
  49. assert result["model"] == "X1C"
  50. @pytest.mark.asyncio
  51. @pytest.mark.integration
  52. async def test_create_printer_with_hostname(self, async_client: AsyncClient):
  53. """Verify printer can be created with a hostname instead of IP address."""
  54. data = {
  55. "name": "DNS Printer",
  56. "serial_number": "00M09A555555555",
  57. "ip_address": "printer.local",
  58. "access_code": "12345678",
  59. "model": "P1S",
  60. }
  61. response = await async_client.post("/api/v1/printers/", json=data)
  62. assert response.status_code == 200
  63. result = response.json()
  64. assert result["name"] == "DNS Printer"
  65. assert result["ip_address"] == "printer.local"
  66. @pytest.mark.asyncio
  67. @pytest.mark.integration
  68. async def test_create_printer_with_fqdn(self, async_client: AsyncClient):
  69. """Verify printer can be created with a fully qualified domain name."""
  70. data = {
  71. "name": "FQDN Printer",
  72. "serial_number": "00M09A666666666",
  73. "ip_address": "my-printer.home.lan",
  74. "access_code": "12345678",
  75. "model": "X1C",
  76. }
  77. response = await async_client.post("/api/v1/printers/", json=data)
  78. assert response.status_code == 200
  79. result = response.json()
  80. assert result["ip_address"] == "my-printer.home.lan"
  81. @pytest.mark.asyncio
  82. @pytest.mark.integration
  83. async def test_create_printer_invalid_hostname(self, async_client: AsyncClient):
  84. """Verify invalid hostnames are rejected."""
  85. data = {
  86. "name": "Bad Printer",
  87. "serial_number": "00M09A777777777",
  88. "ip_address": "-invalid",
  89. "access_code": "12345678",
  90. }
  91. response = await async_client.post("/api/v1/printers/", json=data)
  92. assert response.status_code == 422
  93. @pytest.mark.asyncio
  94. @pytest.mark.integration
  95. async def test_create_printer_duplicate_serial(self, async_client: AsyncClient, printer_factory, db_session):
  96. """Verify duplicate serial number is rejected."""
  97. await printer_factory(serial_number="00M09A222222222")
  98. data = {
  99. "name": "Duplicate Printer",
  100. "serial_number": "00M09A222222222",
  101. "ip_address": "192.168.1.101",
  102. "access_code": "12345678",
  103. }
  104. response = await async_client.post("/api/v1/printers/", json=data)
  105. # Should fail due to duplicate serial
  106. assert response.status_code in [400, 409, 422, 500]
  107. # ========================================================================
  108. # Get single endpoint
  109. # ========================================================================
  110. @pytest.mark.asyncio
  111. @pytest.mark.integration
  112. async def test_get_printer(self, async_client: AsyncClient, printer_factory, db_session):
  113. """Verify single printer can be retrieved."""
  114. printer = await printer_factory(name="Get Test Printer")
  115. response = await async_client.get(f"/api/v1/printers/{printer.id}")
  116. assert response.status_code == 200
  117. result = response.json()
  118. assert result["id"] == printer.id
  119. assert result["name"] == "Get Test Printer"
  120. @pytest.mark.asyncio
  121. @pytest.mark.integration
  122. async def test_get_printer_not_found(self, async_client: AsyncClient):
  123. """Verify 404 for non-existent printer."""
  124. response = await async_client.get("/api/v1/printers/9999")
  125. assert response.status_code == 404
  126. # ========================================================================
  127. # Update endpoints
  128. # ========================================================================
  129. @pytest.mark.asyncio
  130. @pytest.mark.integration
  131. async def test_update_printer_name(self, async_client: AsyncClient, printer_factory, db_session):
  132. """Verify printer name can be updated."""
  133. printer = await printer_factory(name="Original Name")
  134. response = await async_client.patch(f"/api/v1/printers/{printer.id}", json={"name": "Updated Name"})
  135. assert response.status_code == 200
  136. assert response.json()["name"] == "Updated Name"
  137. @pytest.mark.asyncio
  138. @pytest.mark.integration
  139. async def test_update_printer_active_status(self, async_client: AsyncClient, printer_factory, db_session):
  140. """Verify printer active status can be updated."""
  141. printer = await printer_factory(is_active=True)
  142. response = await async_client.patch(f"/api/v1/printers/{printer.id}", json={"is_active": False})
  143. assert response.status_code == 200
  144. assert response.json()["is_active"] is False
  145. @pytest.mark.asyncio
  146. @pytest.mark.integration
  147. async def test_update_printer_auto_archive(self, async_client: AsyncClient, printer_factory, db_session):
  148. """Verify auto_archive setting can be updated."""
  149. printer = await printer_factory(auto_archive=True)
  150. response = await async_client.patch(f"/api/v1/printers/{printer.id}", json={"auto_archive": False})
  151. assert response.status_code == 200
  152. assert response.json()["auto_archive"] is False
  153. @pytest.mark.asyncio
  154. @pytest.mark.integration
  155. async def test_update_nonexistent_printer(self, async_client: AsyncClient):
  156. """Verify updating non-existent printer returns 404."""
  157. response = await async_client.patch("/api/v1/printers/9999", json={"name": "New Name"})
  158. assert response.status_code == 404
  159. # ========================================================================
  160. # Delete endpoints
  161. # ========================================================================
  162. @pytest.mark.asyncio
  163. @pytest.mark.integration
  164. async def test_delete_printer(self, async_client: AsyncClient, printer_factory, db_session):
  165. """Verify printer can be deleted."""
  166. printer = await printer_factory()
  167. printer_id = printer.id
  168. response = await async_client.delete(f"/api/v1/printers/{printer_id}")
  169. assert response.status_code == 200
  170. # Verify deleted
  171. response = await async_client.get(f"/api/v1/printers/{printer_id}")
  172. assert response.status_code == 404
  173. @pytest.mark.asyncio
  174. @pytest.mark.integration
  175. async def test_delete_nonexistent_printer(self, async_client: AsyncClient):
  176. """Verify deleting non-existent printer returns 404."""
  177. response = await async_client.delete("/api/v1/printers/9999")
  178. assert response.status_code == 404
  179. # ========================================================================
  180. # Status endpoint
  181. # ========================================================================
  182. @pytest.mark.asyncio
  183. @pytest.mark.integration
  184. async def test_get_printer_status(
  185. self, async_client: AsyncClient, printer_factory, mock_printer_manager, db_session
  186. ):
  187. """Verify printer status can be retrieved."""
  188. printer = await printer_factory()
  189. response = await async_client.get(f"/api/v1/printers/{printer.id}/status")
  190. assert response.status_code == 200
  191. result = response.json()
  192. assert "connected" in result
  193. assert "state" in result
  194. @pytest.mark.asyncio
  195. @pytest.mark.integration
  196. async def test_get_printer_status_not_found(self, async_client: AsyncClient):
  197. """Verify 404 for status of non-existent printer."""
  198. response = await async_client.get("/api/v1/printers/9999/status")
  199. assert response.status_code == 404
  200. # ========================================================================
  201. # Test connection endpoint
  202. # ========================================================================
  203. class TestPrinterDataIntegrity:
  204. """Tests for printer data integrity."""
  205. @pytest.mark.asyncio
  206. @pytest.mark.integration
  207. async def test_printer_stores_all_fields(self, async_client: AsyncClient, printer_factory, db_session):
  208. """Verify printer stores all fields correctly."""
  209. printer = await printer_factory(
  210. name="Full Test Printer",
  211. serial_number="00M09A444444444",
  212. ip_address="192.168.1.150",
  213. model="P1S",
  214. is_active=True,
  215. auto_archive=False,
  216. )
  217. response = await async_client.get(f"/api/v1/printers/{printer.id}")
  218. assert response.status_code == 200
  219. result = response.json()
  220. assert result["name"] == "Full Test Printer"
  221. assert result["serial_number"] == "00M09A444444444"
  222. assert result["ip_address"] == "192.168.1.150"
  223. assert result["model"] == "P1S"
  224. assert result["is_active"] is True
  225. assert result["auto_archive"] is False
  226. @pytest.mark.asyncio
  227. @pytest.mark.integration
  228. async def test_printer_update_persists(self, async_client: AsyncClient, printer_factory, db_session):
  229. """CRITICAL: Verify printer updates persist."""
  230. printer = await printer_factory(name="Original", is_active=True)
  231. # Update
  232. await async_client.patch(f"/api/v1/printers/{printer.id}", json={"name": "Updated", "is_active": False})
  233. # Verify persistence
  234. response = await async_client.get(f"/api/v1/printers/{printer.id}")
  235. result = response.json()
  236. assert result["name"] == "Updated"
  237. assert result["is_active"] is False
  238. # ========================================================================
  239. # Refresh status endpoint
  240. # ========================================================================
  241. @pytest.mark.asyncio
  242. @pytest.mark.integration
  243. async def test_refresh_status_not_found(self, async_client: AsyncClient):
  244. """Verify 404 for non-existent printer."""
  245. response = await async_client.post("/api/v1/printers/99999/refresh-status")
  246. assert response.status_code == 404
  247. @pytest.mark.asyncio
  248. @pytest.mark.integration
  249. async def test_refresh_status_not_connected(self, async_client: AsyncClient, printer_factory):
  250. """Verify 400 when printer is not connected."""
  251. printer = await printer_factory(name="Disconnected Printer")
  252. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  253. mock_pm.request_status_update.return_value = False
  254. response = await async_client.post(f"/api/v1/printers/{printer.id}/refresh-status")
  255. assert response.status_code == 400
  256. assert "not connected" in response.json()["detail"].lower()
  257. @pytest.mark.asyncio
  258. @pytest.mark.integration
  259. async def test_refresh_status_success(self, async_client: AsyncClient, printer_factory):
  260. """Verify successful refresh request."""
  261. printer = await printer_factory(name="Connected Printer")
  262. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  263. mock_pm.request_status_update.return_value = True
  264. response = await async_client.post(f"/api/v1/printers/{printer.id}/refresh-status")
  265. assert response.status_code == 200
  266. assert response.json()["status"] == "refresh_requested"
  267. mock_pm.request_status_update.assert_called_once_with(printer.id)
  268. # ========================================================================
  269. # Current print user endpoint (Issue #206)
  270. # ========================================================================
  271. @pytest.mark.asyncio
  272. @pytest.mark.integration
  273. async def test_get_current_print_user_not_found(self, async_client: AsyncClient):
  274. """Verify 404 for non-existent printer."""
  275. response = await async_client.get("/api/v1/printers/99999/current-print-user")
  276. assert response.status_code == 404
  277. @pytest.mark.asyncio
  278. @pytest.mark.integration
  279. async def test_get_current_print_user_returns_empty_when_no_user(self, async_client: AsyncClient, printer_factory):
  280. """Verify empty object returned when no user is tracked."""
  281. printer = await printer_factory(name="Test Printer")
  282. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  283. mock_pm.get_current_print_user.return_value = None
  284. response = await async_client.get(f"/api/v1/printers/{printer.id}/current-print-user")
  285. assert response.status_code == 200
  286. assert response.json() == {}
  287. @pytest.mark.asyncio
  288. @pytest.mark.integration
  289. async def test_get_current_print_user_returns_user_info(self, async_client: AsyncClient, printer_factory):
  290. """Verify user info is returned when tracked."""
  291. printer = await printer_factory(name="Test Printer")
  292. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  293. mock_pm.get_current_print_user.return_value = {"user_id": 42, "username": "testuser"}
  294. response = await async_client.get(f"/api/v1/printers/{printer.id}/current-print-user")
  295. assert response.status_code == 200
  296. result = response.json()
  297. assert result["user_id"] == 42
  298. assert result["username"] == "testuser"
  299. class TestPrintControlAPI:
  300. """Integration tests for print control endpoints (stop, pause, resume)."""
  301. # ========================================================================
  302. # Stop print endpoint
  303. # ========================================================================
  304. @pytest.mark.asyncio
  305. @pytest.mark.integration
  306. async def test_stop_print_not_found(self, async_client: AsyncClient):
  307. """Verify 404 for non-existent printer."""
  308. response = await async_client.post("/api/v1/printers/99999/print/stop")
  309. assert response.status_code == 404
  310. @pytest.mark.asyncio
  311. @pytest.mark.integration
  312. async def test_stop_print_not_connected(self, async_client: AsyncClient, printer_factory):
  313. """Verify error when printer is not connected."""
  314. printer = await printer_factory(name="Disconnected Printer")
  315. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  316. mock_pm.get_client.return_value = None
  317. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/stop")
  318. assert response.status_code == 400
  319. assert "not connected" in response.json()["detail"].lower()
  320. @pytest.mark.asyncio
  321. @pytest.mark.integration
  322. async def test_stop_print_success(self, async_client: AsyncClient, printer_factory):
  323. """Verify successful stop print request."""
  324. printer = await printer_factory(name="Printing Printer")
  325. mock_client = MagicMock()
  326. mock_client.stop_print.return_value = True
  327. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  328. mock_pm.get_client.return_value = mock_client
  329. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/stop")
  330. assert response.status_code == 200
  331. assert response.json()["success"] is True
  332. mock_client.stop_print.assert_called_once()
  333. # ========================================================================
  334. # Pause print endpoint
  335. # ========================================================================
  336. @pytest.mark.asyncio
  337. @pytest.mark.integration
  338. async def test_pause_print_not_found(self, async_client: AsyncClient):
  339. """Verify 404 for non-existent printer."""
  340. response = await async_client.post("/api/v1/printers/99999/print/pause")
  341. assert response.status_code == 404
  342. @pytest.mark.asyncio
  343. @pytest.mark.integration
  344. async def test_pause_print_not_connected(self, async_client: AsyncClient, printer_factory):
  345. """Verify error when printer is not connected."""
  346. printer = await printer_factory(name="Disconnected Printer")
  347. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  348. mock_pm.get_client.return_value = None
  349. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/pause")
  350. assert response.status_code == 400
  351. assert "not connected" in response.json()["detail"].lower()
  352. @pytest.mark.asyncio
  353. @pytest.mark.integration
  354. async def test_pause_print_success(self, async_client: AsyncClient, printer_factory):
  355. """Verify successful pause print request."""
  356. printer = await printer_factory(name="Printing Printer")
  357. mock_client = MagicMock()
  358. mock_client.pause_print.return_value = True
  359. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  360. mock_pm.get_client.return_value = mock_client
  361. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/pause")
  362. assert response.status_code == 200
  363. assert response.json()["success"] is True
  364. mock_client.pause_print.assert_called_once()
  365. # ========================================================================
  366. # Resume print endpoint
  367. # ========================================================================
  368. @pytest.mark.asyncio
  369. @pytest.mark.integration
  370. async def test_resume_print_not_found(self, async_client: AsyncClient):
  371. """Verify 404 for non-existent printer."""
  372. response = await async_client.post("/api/v1/printers/99999/print/resume")
  373. assert response.status_code == 404
  374. @pytest.mark.asyncio
  375. @pytest.mark.integration
  376. async def test_resume_print_not_connected(self, async_client: AsyncClient, printer_factory):
  377. """Verify error when printer is not connected."""
  378. printer = await printer_factory(name="Disconnected Printer")
  379. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  380. mock_pm.get_client.return_value = None
  381. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/resume")
  382. assert response.status_code == 400
  383. assert "not connected" in response.json()["detail"].lower()
  384. @pytest.mark.asyncio
  385. @pytest.mark.integration
  386. async def test_resume_print_success(self, async_client: AsyncClient, printer_factory):
  387. """Verify successful resume print request."""
  388. printer = await printer_factory(name="Paused Printer")
  389. mock_client = MagicMock()
  390. mock_client.resume_print.return_value = True
  391. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  392. mock_pm.get_client.return_value = mock_client
  393. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/resume")
  394. assert response.status_code == 200
  395. assert response.json()["success"] is True
  396. mock_client.resume_print.assert_called_once()
  397. class TestAMSRefreshAPI:
  398. """Integration tests for AMS slot refresh endpoint."""
  399. @pytest.mark.asyncio
  400. @pytest.mark.integration
  401. async def test_ams_refresh_not_found(self, async_client: AsyncClient):
  402. """Verify 404 for non-existent printer."""
  403. response = await async_client.post("/api/v1/printers/99999/ams/0/slot/0/refresh")
  404. assert response.status_code == 404
  405. @pytest.mark.asyncio
  406. @pytest.mark.integration
  407. async def test_ams_refresh_not_connected(self, async_client: AsyncClient, printer_factory):
  408. """Verify error when printer is not connected."""
  409. printer = await printer_factory(name="Disconnected Printer")
  410. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  411. mock_pm.get_client.return_value = None
  412. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/0/slot/0/refresh")
  413. assert response.status_code == 400
  414. assert "not connected" in response.json()["detail"].lower()
  415. @pytest.mark.asyncio
  416. @pytest.mark.integration
  417. async def test_ams_refresh_success(self, async_client: AsyncClient, printer_factory):
  418. """Verify successful AMS refresh request."""
  419. printer = await printer_factory(name="Printer with AMS")
  420. mock_client = MagicMock()
  421. mock_client.ams_refresh_tray.return_value = (True, "Refreshing AMS 0 tray 1")
  422. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  423. mock_pm.get_client.return_value = mock_client
  424. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/0/slot/1/refresh")
  425. assert response.status_code == 200
  426. result = response.json()
  427. assert result["success"] is True
  428. mock_client.ams_refresh_tray.assert_called_once_with(0, 1)
  429. @pytest.mark.asyncio
  430. @pytest.mark.integration
  431. async def test_ams_refresh_filament_loaded(self, async_client: AsyncClient, printer_factory):
  432. """Verify error when filament is loaded (can't refresh while loaded)."""
  433. printer = await printer_factory(name="Printer with AMS")
  434. mock_client = MagicMock()
  435. mock_client.ams_refresh_tray.return_value = (False, "Please unload filament first")
  436. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  437. mock_pm.get_client.return_value = mock_client
  438. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/0/slot/0/refresh")
  439. assert response.status_code == 400
  440. assert "unload" in response.json()["detail"].lower()
  441. class TestSkipObjectsAPI:
  442. """Integration tests for skip objects endpoints."""
  443. # ========================================================================
  444. # Get printable objects endpoint
  445. # ========================================================================
  446. @pytest.mark.asyncio
  447. @pytest.mark.integration
  448. async def test_get_objects_not_found(self, async_client: AsyncClient):
  449. """Verify 404 for non-existent printer."""
  450. response = await async_client.get("/api/v1/printers/99999/print/objects")
  451. assert response.status_code == 404
  452. @pytest.mark.asyncio
  453. @pytest.mark.integration
  454. async def test_get_objects_not_connected(self, async_client: AsyncClient, printer_factory):
  455. """Verify error when printer is not connected."""
  456. printer = await printer_factory(name="Disconnected Printer")
  457. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  458. mock_pm.get_client.return_value = None
  459. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  460. assert response.status_code == 400
  461. assert "not connected" in response.json()["detail"].lower()
  462. @pytest.mark.asyncio
  463. @pytest.mark.integration
  464. async def test_get_objects_empty(self, async_client: AsyncClient, printer_factory):
  465. """Verify empty objects list when no print is active."""
  466. printer = await printer_factory(name="Idle Printer")
  467. mock_client = MagicMock()
  468. mock_client.state.printable_objects = {}
  469. mock_client.state.skipped_objects = []
  470. mock_client.state.state = "IDLE"
  471. mock_client.state.subtask_name = None # Prevent FTP download attempt
  472. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  473. mock_pm.get_client.return_value = mock_client
  474. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  475. assert response.status_code == 200
  476. result = response.json()
  477. assert result["objects"] == []
  478. assert result["total"] == 0
  479. assert result["skipped_count"] == 0
  480. assert result["is_printing"] is False
  481. @pytest.mark.asyncio
  482. @pytest.mark.integration
  483. async def test_get_objects_with_data(self, async_client: AsyncClient, printer_factory):
  484. """Verify objects list when print is active."""
  485. printer = await printer_factory(name="Printing Printer")
  486. mock_client = MagicMock()
  487. mock_client.state.printable_objects = {100: "Part A", 200: "Part B", 300: "Part C"}
  488. mock_client.state.skipped_objects = [200]
  489. mock_client.state.state = "RUNNING"
  490. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  491. mock_pm.get_client.return_value = mock_client
  492. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  493. assert response.status_code == 200
  494. result = response.json()
  495. assert result["total"] == 3
  496. assert result["skipped_count"] == 1
  497. assert result["is_printing"] is True
  498. # Check objects have correct structure
  499. objects_by_id = {obj["id"]: obj for obj in result["objects"]}
  500. assert objects_by_id[100]["name"] == "Part A"
  501. assert objects_by_id[100]["skipped"] is False
  502. assert objects_by_id[200]["name"] == "Part B"
  503. assert objects_by_id[200]["skipped"] is True
  504. assert objects_by_id[300]["name"] == "Part C"
  505. assert objects_by_id[300]["skipped"] is False
  506. # ========================================================================
  507. # Skip objects endpoint
  508. # ========================================================================
  509. @pytest.mark.asyncio
  510. @pytest.mark.integration
  511. async def test_get_objects_with_positions(self, async_client: AsyncClient, printer_factory):
  512. """Verify objects list includes position data when available."""
  513. printer = await printer_factory(name="Printing Printer")
  514. # New format with position data
  515. mock_client = MagicMock()
  516. mock_client.state.printable_objects = {
  517. 100: {"name": "Part A", "x": 50.0, "y": 100.0},
  518. 200: {"name": "Part B", "x": 150.0, "y": 100.0},
  519. }
  520. mock_client.state.skipped_objects = []
  521. mock_client.state.state = "RUNNING"
  522. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  523. mock_pm.get_client.return_value = mock_client
  524. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  525. assert response.status_code == 200
  526. result = response.json()
  527. assert result["total"] == 2
  528. # Check objects have position data
  529. objects_by_id = {obj["id"]: obj for obj in result["objects"]}
  530. assert objects_by_id[100]["name"] == "Part A"
  531. assert objects_by_id[100]["x"] == 50.0
  532. assert objects_by_id[100]["y"] == 100.0
  533. assert objects_by_id[200]["name"] == "Part B"
  534. assert objects_by_id[200]["x"] == 150.0
  535. assert objects_by_id[200]["y"] == 100.0
  536. # ========================================================================
  537. # Skip objects endpoint
  538. # ========================================================================
  539. @pytest.mark.asyncio
  540. @pytest.mark.integration
  541. async def test_skip_objects_not_found(self, async_client: AsyncClient):
  542. """Verify 404 for non-existent printer."""
  543. response = await async_client.post("/api/v1/printers/99999/print/skip-objects", json=[100])
  544. assert response.status_code == 404
  545. @pytest.mark.asyncio
  546. @pytest.mark.integration
  547. async def test_skip_objects_not_connected(self, async_client: AsyncClient, printer_factory):
  548. """Verify error when printer is not connected."""
  549. printer = await printer_factory(name="Disconnected Printer")
  550. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  551. mock_pm.get_client.return_value = None
  552. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[100])
  553. assert response.status_code == 400
  554. assert "not connected" in response.json()["detail"].lower()
  555. @pytest.mark.asyncio
  556. @pytest.mark.integration
  557. async def test_skip_objects_empty_list(self, async_client: AsyncClient, printer_factory):
  558. """Verify error when no object IDs provided."""
  559. printer = await printer_factory(name="Printing Printer")
  560. mock_client = MagicMock()
  561. mock_client.state.printable_objects = {100: "Part A"}
  562. mock_client.state.skipped_objects = []
  563. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  564. mock_pm.get_client.return_value = mock_client
  565. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[])
  566. assert response.status_code == 400
  567. assert "no object" in response.json()["detail"].lower()
  568. @pytest.mark.asyncio
  569. @pytest.mark.integration
  570. async def test_skip_objects_invalid_id(self, async_client: AsyncClient, printer_factory):
  571. """Verify error when object ID doesn't exist."""
  572. printer = await printer_factory(name="Printing Printer")
  573. mock_client = MagicMock()
  574. mock_client.state.printable_objects = {100: "Part A"}
  575. mock_client.state.skipped_objects = []
  576. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  577. mock_pm.get_client.return_value = mock_client
  578. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[999])
  579. assert response.status_code == 400
  580. assert "invalid" in response.json()["detail"].lower()
  581. @pytest.mark.asyncio
  582. @pytest.mark.integration
  583. async def test_skip_objects_success(self, async_client: AsyncClient, printer_factory):
  584. """Verify successful skip objects request."""
  585. printer = await printer_factory(name="Printing Printer")
  586. mock_client = MagicMock()
  587. mock_client.state.printable_objects = {100: "Part A", 200: "Part B"}
  588. mock_client.state.skipped_objects = []
  589. mock_client.skip_objects.return_value = True
  590. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  591. mock_pm.get_client.return_value = mock_client
  592. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[100])
  593. assert response.status_code == 200
  594. result = response.json()
  595. assert result["success"] is True
  596. assert 100 in result["skipped_objects"]
  597. mock_client.skip_objects.assert_called_once_with([100])
  598. @pytest.mark.asyncio
  599. @pytest.mark.integration
  600. async def test_skip_objects_multiple(self, async_client: AsyncClient, printer_factory):
  601. """Verify skipping multiple objects at once."""
  602. printer = await printer_factory(name="Printing Printer")
  603. mock_client = MagicMock()
  604. mock_client.state.printable_objects = {100: "Part A", 200: "Part B", 300: "Part C"}
  605. mock_client.state.skipped_objects = []
  606. mock_client.skip_objects.return_value = True
  607. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  608. mock_pm.get_client.return_value = mock_client
  609. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[100, 200])
  610. assert response.status_code == 200
  611. result = response.json()
  612. assert result["success"] is True
  613. assert 100 in result["skipped_objects"]
  614. assert 200 in result["skipped_objects"]
  615. mock_client.skip_objects.assert_called_once_with([100, 200])
  616. class TestChamberLightAPI:
  617. """Integration tests for chamber light control endpoint."""
  618. @pytest.mark.asyncio
  619. @pytest.mark.integration
  620. async def test_chamber_light_not_found(self, async_client: AsyncClient):
  621. """Verify 404 for non-existent printer."""
  622. response = await async_client.post("/api/v1/printers/99999/chamber-light?on=true")
  623. assert response.status_code == 404
  624. @pytest.mark.asyncio
  625. @pytest.mark.integration
  626. async def test_chamber_light_not_connected(self, async_client: AsyncClient, printer_factory):
  627. """Verify error when printer is not connected."""
  628. printer = await printer_factory(name="Disconnected Printer")
  629. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  630. mock_pm.get_client.return_value = None
  631. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=true")
  632. assert response.status_code == 400
  633. assert "not connected" in response.json()["detail"].lower()
  634. @pytest.mark.asyncio
  635. @pytest.mark.integration
  636. async def test_chamber_light_on_success(self, async_client: AsyncClient, printer_factory):
  637. """Verify successful chamber light on request."""
  638. printer = await printer_factory(name="Test Printer")
  639. mock_client = MagicMock()
  640. mock_client.set_chamber_light.return_value = True
  641. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  642. mock_pm.get_client.return_value = mock_client
  643. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=true")
  644. assert response.status_code == 200
  645. result = response.json()
  646. assert result["success"] is True
  647. assert "on" in result["message"].lower()
  648. mock_client.set_chamber_light.assert_called_once_with(True)
  649. @pytest.mark.asyncio
  650. @pytest.mark.integration
  651. async def test_chamber_light_off_success(self, async_client: AsyncClient, printer_factory):
  652. """Verify successful chamber light off request."""
  653. printer = await printer_factory(name="Test Printer")
  654. mock_client = MagicMock()
  655. mock_client.set_chamber_light.return_value = True
  656. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  657. mock_pm.get_client.return_value = mock_client
  658. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=false")
  659. assert response.status_code == 200
  660. result = response.json()
  661. assert result["success"] is True
  662. assert "off" in result["message"].lower()
  663. mock_client.set_chamber_light.assert_called_once_with(False)
  664. @pytest.mark.asyncio
  665. @pytest.mark.integration
  666. async def test_chamber_light_failure(self, async_client: AsyncClient, printer_factory):
  667. """Verify error handling when chamber light control fails."""
  668. printer = await printer_factory(name="Test Printer")
  669. mock_client = MagicMock()
  670. mock_client.set_chamber_light.return_value = False
  671. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  672. mock_pm.get_client.return_value = mock_client
  673. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=true")
  674. assert response.status_code == 500
  675. assert "failed" in response.json()["detail"].lower()
  676. class TestClearHMSErrorsAPI:
  677. """Integration tests for clear HMS errors endpoint."""
  678. @pytest.mark.asyncio
  679. @pytest.mark.integration
  680. async def test_clear_hms_errors_not_found(self, async_client: AsyncClient):
  681. """Verify 404 for non-existent printer."""
  682. response = await async_client.post("/api/v1/printers/99999/hms/clear")
  683. assert response.status_code == 404
  684. @pytest.mark.asyncio
  685. @pytest.mark.integration
  686. async def test_clear_hms_errors_not_connected(self, async_client: AsyncClient, printer_factory):
  687. """Verify error when printer is not connected."""
  688. printer = await printer_factory(name="Disconnected Printer")
  689. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  690. mock_pm.get_client.return_value = None
  691. response = await async_client.post(f"/api/v1/printers/{printer.id}/hms/clear")
  692. assert response.status_code == 400
  693. assert "not connected" in response.json()["detail"].lower()
  694. @pytest.mark.asyncio
  695. @pytest.mark.integration
  696. async def test_clear_hms_errors_success(self, async_client: AsyncClient, printer_factory):
  697. """Verify successful clear HMS errors request."""
  698. printer = await printer_factory(name="Test Printer")
  699. mock_client = MagicMock()
  700. mock_client.clear_hms_errors.return_value = True
  701. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  702. mock_pm.get_client.return_value = mock_client
  703. response = await async_client.post(f"/api/v1/printers/{printer.id}/hms/clear")
  704. assert response.status_code == 200
  705. result = response.json()
  706. assert result["success"] is True
  707. assert "cleared" in result["message"].lower()
  708. mock_client.clear_hms_errors.assert_called_once()
  709. @pytest.mark.asyncio
  710. @pytest.mark.integration
  711. async def test_clear_hms_errors_failure(self, async_client: AsyncClient, printer_factory):
  712. """Verify error handling when clear HMS errors fails."""
  713. printer = await printer_factory(name="Test Printer")
  714. mock_client = MagicMock()
  715. mock_client.clear_hms_errors.return_value = False
  716. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  717. mock_pm.get_client.return_value = mock_client
  718. response = await async_client.post(f"/api/v1/printers/{printer.id}/hms/clear")
  719. assert response.status_code == 500
  720. assert "failed" in response.json()["detail"].lower()