test_printers_api.py 66 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580
  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. @pytest.mark.asyncio
  201. @pytest.mark.integration
  202. async def test_get_printer_status_includes_fila_switch_when_installed(
  203. self, async_client: AsyncClient, printer_factory, db_session
  204. ):
  205. """When the FTS accessory is installed, the status response must include
  206. the fila_switch object with the routing arrays. See #1162.
  207. The accessory is detected from print.device.fila_switch in MQTT;
  208. we feed a PrinterState with FilaSwitchState(installed=True, ...) and
  209. confirm it survives the schema serialization round-trip.
  210. """
  211. from unittest.mock import MagicMock, patch
  212. from backend.app.services.bambu_mqtt import FilaSwitchState, PrinterState
  213. printer = await printer_factory()
  214. state = PrinterState()
  215. state.connected = True
  216. state.state = "IDLE"
  217. state.fila_switch = FilaSwitchState(
  218. installed=True,
  219. in_slots=[-1, 2],
  220. out_extruders=[0, 1],
  221. stat=0,
  222. info=2,
  223. )
  224. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  225. mock_pm.get_status = MagicMock(return_value=state)
  226. mock_pm.is_awaiting_plate_clear = MagicMock(return_value=False)
  227. response = await async_client.get(f"/api/v1/printers/{printer.id}/status")
  228. assert response.status_code == 200
  229. result = response.json()
  230. assert result["fila_switch"] is not None
  231. assert result["fila_switch"]["installed"] is True
  232. assert result["fila_switch"]["in_slots"] == [-1, 2]
  233. assert result["fila_switch"]["out_extruders"] == [0, 1]
  234. assert result["fila_switch"]["stat"] == 0
  235. assert result["fila_switch"]["info"] == 2
  236. @pytest.mark.asyncio
  237. @pytest.mark.integration
  238. async def test_cover_uses_dispatched_plate_when_gcode_file_lacks_path(
  239. self, async_client: AsyncClient, printer_factory, db_session, tmp_path
  240. ):
  241. """When firmware drops the plate path from gcode_file (e.g. P1S
  242. 01.10.00.00, #1166), the dispatched-plate record must take precedence
  243. and serve plate 4's thumbnail instead of falling back to plate_1.png."""
  244. import io
  245. import zipfile
  246. from unittest.mock import MagicMock, patch
  247. from backend.app.services.bambu_ftp import cache_3mf_download
  248. from backend.app.services.bambu_mqtt import PrinterState
  249. printer = await printer_factory()
  250. # Build a 3MF that mimics a "true" multi-plate archive: thumbnails
  251. # for plates 1..4 are all present, gcode files for plates 1..4 are
  252. # all present. Without the dispatch record we'd default to plate_1.png.
  253. threemf_path = tmp_path / "MyModel.3mf"
  254. with zipfile.ZipFile(threemf_path, "w") as zf:
  255. for plate in range(1, 5):
  256. zf.writestr(f"Metadata/plate_{plate}.png", f"PLATE_{plate}_PNG".encode())
  257. zf.writestr(f"Metadata/plate_{plate}.gcode", f"; plate {plate} gcode\n")
  258. cache_3mf_download(printer.id, "MyModel.3mf", threemf_path)
  259. state = PrinterState()
  260. state.connected = True
  261. state.state = "RUNNING"
  262. state.subtask_name = "MyModel"
  263. state.gcode_file = "MyModel.3mf" # firmware drops plate path
  264. state.dispatched_plate_id = 4
  265. state.dispatched_subtask = "MyModel"
  266. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  267. mock_pm.get_status = MagicMock(return_value=state)
  268. mock_pm.is_awaiting_plate_clear = MagicMock(return_value=False)
  269. response = await async_client.get(f"/api/v1/printers/{printer.id}/cover")
  270. assert response.status_code == 200
  271. assert response.content == b"PLATE_4_PNG"
  272. @pytest.mark.asyncio
  273. @pytest.mark.integration
  274. async def test_cover_3mf_scan_fallback_for_per_plate_archive(
  275. self, async_client: AsyncClient, printer_factory, db_session, tmp_path
  276. ):
  277. """Per-plate archives sliced separately in Bambu Studio contain a
  278. single Metadata/plate_N.gcode (the active plate) but bundle thumbnails
  279. for every plate. With no dispatch record (e.g. dispatched via Studio
  280. directly) and no plate path in gcode_file, the route must scan the
  281. 3MF and pick plate N's thumbnail. See #1166 option 4."""
  282. import zipfile
  283. from unittest.mock import MagicMock, patch
  284. from backend.app.services.bambu_ftp import cache_3mf_download
  285. from backend.app.services.bambu_mqtt import PrinterState
  286. printer = await printer_factory()
  287. # Per-plate archive: thumbnails for all plates, gcode for plate 3 only.
  288. threemf_path = tmp_path / "PerPlate.3mf"
  289. with zipfile.ZipFile(threemf_path, "w") as zf:
  290. for plate in range(1, 5):
  291. zf.writestr(f"Metadata/plate_{plate}.png", f"PLATE_{plate}_PNG".encode())
  292. zf.writestr("Metadata/plate_3.gcode", "; only plate 3 has gcode\n")
  293. cache_3mf_download(printer.id, "PerPlate.3mf", threemf_path)
  294. state = PrinterState()
  295. state.connected = True
  296. state.state = "RUNNING"
  297. state.subtask_name = "PerPlate"
  298. state.gcode_file = "PerPlate.3mf"
  299. # No dispatch record (Studio-direct dispatch).
  300. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  301. mock_pm.get_status = MagicMock(return_value=state)
  302. mock_pm.is_awaiting_plate_clear = MagicMock(return_value=False)
  303. response = await async_client.get(f"/api/v1/printers/{printer.id}/cover")
  304. assert response.status_code == 200
  305. assert response.content == b"PLATE_3_PNG"
  306. @pytest.mark.asyncio
  307. @pytest.mark.integration
  308. async def test_get_printer_status_omits_fila_switch_when_not_installed(
  309. self, async_client: AsyncClient, printer_factory, db_session
  310. ):
  311. """Without the FTS accessory, fila_switch must be null so the frontend
  312. keeps applying the per-extruder filter on regular dual-nozzle printers."""
  313. from unittest.mock import MagicMock, patch
  314. from backend.app.services.bambu_mqtt import PrinterState
  315. printer = await printer_factory()
  316. state = PrinterState()
  317. state.connected = True
  318. state.state = "IDLE"
  319. # default fila_switch — installed = False
  320. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  321. mock_pm.get_status = MagicMock(return_value=state)
  322. mock_pm.is_awaiting_plate_clear = MagicMock(return_value=False)
  323. response = await async_client.get(f"/api/v1/printers/{printer.id}/status")
  324. assert response.status_code == 200
  325. result = response.json()
  326. assert result["fila_switch"] is None
  327. # ========================================================================
  328. # Test connection endpoint
  329. # ========================================================================
  330. class TestPrinterDataIntegrity:
  331. """Tests for printer data integrity."""
  332. @pytest.mark.asyncio
  333. @pytest.mark.integration
  334. async def test_printer_stores_all_fields(self, async_client: AsyncClient, printer_factory, db_session):
  335. """Verify printer stores all fields correctly."""
  336. printer = await printer_factory(
  337. name="Full Test Printer",
  338. serial_number="00M09A444444444",
  339. ip_address="192.168.1.150",
  340. model="P1S",
  341. is_active=True,
  342. auto_archive=False,
  343. )
  344. response = await async_client.get(f"/api/v1/printers/{printer.id}")
  345. assert response.status_code == 200
  346. result = response.json()
  347. assert result["name"] == "Full Test Printer"
  348. assert result["serial_number"] == "00M09A444444444"
  349. assert result["ip_address"] == "192.168.1.150"
  350. assert result["model"] == "P1S"
  351. assert result["is_active"] is True
  352. assert result["auto_archive"] is False
  353. @pytest.mark.asyncio
  354. @pytest.mark.integration
  355. async def test_printer_update_persists(self, async_client: AsyncClient, printer_factory, db_session):
  356. """CRITICAL: Verify printer updates persist."""
  357. printer = await printer_factory(name="Original", is_active=True)
  358. # Update
  359. await async_client.patch(f"/api/v1/printers/{printer.id}", json={"name": "Updated", "is_active": False})
  360. # Verify persistence
  361. response = await async_client.get(f"/api/v1/printers/{printer.id}")
  362. result = response.json()
  363. assert result["name"] == "Updated"
  364. assert result["is_active"] is False
  365. # ========================================================================
  366. # Refresh status endpoint
  367. # ========================================================================
  368. @pytest.mark.asyncio
  369. @pytest.mark.integration
  370. async def test_refresh_status_not_found(self, async_client: AsyncClient):
  371. """Verify 404 for non-existent printer."""
  372. response = await async_client.post("/api/v1/printers/99999/refresh-status")
  373. assert response.status_code == 404
  374. @pytest.mark.asyncio
  375. @pytest.mark.integration
  376. async def test_refresh_status_not_connected(self, async_client: AsyncClient, printer_factory):
  377. """Verify 400 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.request_status_update.return_value = False
  381. response = await async_client.post(f"/api/v1/printers/{printer.id}/refresh-status")
  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_refresh_status_success(self, async_client: AsyncClient, printer_factory):
  387. """Verify successful refresh request."""
  388. printer = await printer_factory(name="Connected Printer")
  389. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  390. mock_pm.request_status_update.return_value = True
  391. response = await async_client.post(f"/api/v1/printers/{printer.id}/refresh-status")
  392. assert response.status_code == 200
  393. assert response.json()["status"] == "refresh_requested"
  394. mock_pm.request_status_update.assert_called_once_with(printer.id)
  395. # ========================================================================
  396. # Current print user endpoint (Issue #206)
  397. # ========================================================================
  398. @pytest.mark.asyncio
  399. @pytest.mark.integration
  400. async def test_get_current_print_user_not_found(self, async_client: AsyncClient):
  401. """Verify 404 for non-existent printer."""
  402. response = await async_client.get("/api/v1/printers/99999/current-print-user")
  403. assert response.status_code == 404
  404. @pytest.mark.asyncio
  405. @pytest.mark.integration
  406. async def test_get_current_print_user_returns_empty_when_no_user(self, async_client: AsyncClient, printer_factory):
  407. """Verify empty object returned when no user is tracked."""
  408. printer = await printer_factory(name="Test Printer")
  409. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  410. mock_pm.get_current_print_user.return_value = None
  411. response = await async_client.get(f"/api/v1/printers/{printer.id}/current-print-user")
  412. assert response.status_code == 200
  413. assert response.json() == {}
  414. @pytest.mark.asyncio
  415. @pytest.mark.integration
  416. async def test_get_current_print_user_returns_user_info(self, async_client: AsyncClient, printer_factory):
  417. """Verify user info is returned when tracked."""
  418. printer = await printer_factory(name="Test Printer")
  419. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  420. mock_pm.get_current_print_user.return_value = {"user_id": 42, "username": "testuser"}
  421. response = await async_client.get(f"/api/v1/printers/{printer.id}/current-print-user")
  422. assert response.status_code == 200
  423. result = response.json()
  424. assert result["user_id"] == 42
  425. assert result["username"] == "testuser"
  426. class TestPrintControlAPI:
  427. """Integration tests for print control endpoints (stop, pause, resume)."""
  428. # ========================================================================
  429. # Stop print endpoint
  430. # ========================================================================
  431. @pytest.mark.asyncio
  432. @pytest.mark.integration
  433. async def test_stop_print_not_found(self, async_client: AsyncClient):
  434. """Verify 404 for non-existent printer."""
  435. response = await async_client.post("/api/v1/printers/99999/print/stop")
  436. assert response.status_code == 404
  437. @pytest.mark.asyncio
  438. @pytest.mark.integration
  439. async def test_stop_print_not_connected(self, async_client: AsyncClient, printer_factory):
  440. """Verify error when printer is not connected."""
  441. printer = await printer_factory(name="Disconnected Printer")
  442. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  443. mock_pm.get_client.return_value = None
  444. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/stop")
  445. assert response.status_code == 400
  446. assert "not connected" in response.json()["detail"].lower()
  447. @pytest.mark.asyncio
  448. @pytest.mark.integration
  449. async def test_stop_print_success(self, async_client: AsyncClient, printer_factory):
  450. """Verify successful stop print request."""
  451. printer = await printer_factory(name="Printing Printer")
  452. mock_client = MagicMock()
  453. mock_client.stop_print.return_value = True
  454. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  455. mock_pm.get_client.return_value = mock_client
  456. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/stop")
  457. assert response.status_code == 200
  458. assert response.json()["success"] is True
  459. mock_client.stop_print.assert_called_once()
  460. # ========================================================================
  461. # Pause print endpoint
  462. # ========================================================================
  463. @pytest.mark.asyncio
  464. @pytest.mark.integration
  465. async def test_pause_print_not_found(self, async_client: AsyncClient):
  466. """Verify 404 for non-existent printer."""
  467. response = await async_client.post("/api/v1/printers/99999/print/pause")
  468. assert response.status_code == 404
  469. @pytest.mark.asyncio
  470. @pytest.mark.integration
  471. async def test_pause_print_not_connected(self, async_client: AsyncClient, printer_factory):
  472. """Verify error when printer is not connected."""
  473. printer = await printer_factory(name="Disconnected Printer")
  474. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  475. mock_pm.get_client.return_value = None
  476. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/pause")
  477. assert response.status_code == 400
  478. assert "not connected" in response.json()["detail"].lower()
  479. @pytest.mark.asyncio
  480. @pytest.mark.integration
  481. async def test_pause_print_success(self, async_client: AsyncClient, printer_factory):
  482. """Verify successful pause print request."""
  483. printer = await printer_factory(name="Printing Printer")
  484. mock_client = MagicMock()
  485. mock_client.pause_print.return_value = True
  486. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  487. mock_pm.get_client.return_value = mock_client
  488. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/pause")
  489. assert response.status_code == 200
  490. assert response.json()["success"] is True
  491. mock_client.pause_print.assert_called_once()
  492. # ========================================================================
  493. # Resume print endpoint
  494. # ========================================================================
  495. @pytest.mark.asyncio
  496. @pytest.mark.integration
  497. async def test_resume_print_not_found(self, async_client: AsyncClient):
  498. """Verify 404 for non-existent printer."""
  499. response = await async_client.post("/api/v1/printers/99999/print/resume")
  500. assert response.status_code == 404
  501. @pytest.mark.asyncio
  502. @pytest.mark.integration
  503. async def test_resume_print_not_connected(self, async_client: AsyncClient, printer_factory):
  504. """Verify error when printer is not connected."""
  505. printer = await printer_factory(name="Disconnected Printer")
  506. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  507. mock_pm.get_client.return_value = None
  508. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/resume")
  509. assert response.status_code == 400
  510. assert "not connected" in response.json()["detail"].lower()
  511. @pytest.mark.asyncio
  512. @pytest.mark.integration
  513. async def test_resume_print_success(self, async_client: AsyncClient, printer_factory):
  514. """Verify successful resume print request."""
  515. printer = await printer_factory(name="Paused Printer")
  516. mock_client = MagicMock()
  517. mock_client.resume_print.return_value = True
  518. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  519. mock_pm.get_client.return_value = mock_client
  520. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/resume")
  521. assert response.status_code == 200
  522. assert response.json()["success"] is True
  523. mock_client.resume_print.assert_called_once()
  524. class TestAMSRefreshAPI:
  525. """Integration tests for AMS slot refresh endpoint."""
  526. @pytest.mark.asyncio
  527. @pytest.mark.integration
  528. async def test_ams_refresh_not_found(self, async_client: AsyncClient):
  529. """Verify 404 for non-existent printer."""
  530. response = await async_client.post("/api/v1/printers/99999/ams/0/slot/0/refresh")
  531. assert response.status_code == 404
  532. @pytest.mark.asyncio
  533. @pytest.mark.integration
  534. async def test_ams_refresh_not_connected(self, async_client: AsyncClient, printer_factory):
  535. """Verify error when printer is not connected."""
  536. printer = await printer_factory(name="Disconnected Printer")
  537. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  538. mock_pm.get_client.return_value = None
  539. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/0/slot/0/refresh")
  540. assert response.status_code == 400
  541. assert "not connected" in response.json()["detail"].lower()
  542. @pytest.mark.asyncio
  543. @pytest.mark.integration
  544. async def test_ams_refresh_success(self, async_client: AsyncClient, printer_factory):
  545. """Verify successful AMS refresh request."""
  546. printer = await printer_factory(name="Printer with AMS")
  547. mock_client = MagicMock()
  548. mock_client.ams_refresh_tray.return_value = (True, "Refreshing AMS 0 tray 1")
  549. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  550. mock_pm.get_client.return_value = mock_client
  551. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/0/slot/1/refresh")
  552. assert response.status_code == 200
  553. result = response.json()
  554. assert result["success"] is True
  555. mock_client.ams_refresh_tray.assert_called_once_with(0, 1)
  556. @pytest.mark.asyncio
  557. @pytest.mark.integration
  558. async def test_ams_refresh_filament_loaded(self, async_client: AsyncClient, printer_factory):
  559. """Verify error when filament is loaded (can't refresh while loaded)."""
  560. printer = await printer_factory(name="Printer with AMS")
  561. mock_client = MagicMock()
  562. mock_client.ams_refresh_tray.return_value = (False, "Please unload filament first")
  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}/ams/0/slot/0/refresh")
  566. assert response.status_code == 400
  567. assert "unload" in response.json()["detail"].lower()
  568. class TestAMSLoadUnloadAPI:
  569. """Integration tests for AMS load / unload endpoints (#891)."""
  570. # ── load ─────────────────────────────────────────────────────────────────
  571. @pytest.mark.asyncio
  572. @pytest.mark.integration
  573. async def test_load_invalid_tray_id(self, async_client: AsyncClient, printer_factory):
  574. """tray_id outside {0..15, 254, 255} is rejected."""
  575. printer = await printer_factory(name="P")
  576. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=99")
  577. assert response.status_code == 400
  578. @pytest.mark.asyncio
  579. @pytest.mark.integration
  580. async def test_load_not_found(self, async_client: AsyncClient):
  581. response = await async_client.post("/api/v1/printers/99999/ams/load?tray_id=0")
  582. assert response.status_code == 404
  583. @pytest.mark.asyncio
  584. @pytest.mark.integration
  585. async def test_load_not_connected(self, async_client: AsyncClient, printer_factory):
  586. printer = await printer_factory(name="Disconnected")
  587. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  588. mock_pm.get_client.return_value = None
  589. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=0")
  590. assert response.status_code == 400
  591. assert "not connected" in response.json()["detail"].lower()
  592. @pytest.mark.asyncio
  593. @pytest.mark.integration
  594. async def test_load_ams_slot_success(self, async_client: AsyncClient, printer_factory):
  595. """tray_id=5 → AMS 1 slot 2 (1-indexed in the message)."""
  596. printer = await printer_factory(name="P")
  597. mock_client = MagicMock()
  598. mock_client.ams_load_filament.return_value = True
  599. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  600. mock_pm.get_client.return_value = mock_client
  601. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=5")
  602. assert response.status_code == 200
  603. mock_client.ams_load_filament.assert_called_once_with(5)
  604. assert "AMS 1" in response.json()["message"]
  605. @pytest.mark.asyncio
  606. @pytest.mark.integration
  607. async def test_load_external_left_success(self, async_client: AsyncClient, printer_factory):
  608. """tray_id=254 → external spool / Ext-L."""
  609. printer = await printer_factory(name="P")
  610. mock_client = MagicMock()
  611. mock_client.ams_load_filament.return_value = True
  612. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  613. mock_pm.get_client.return_value = mock_client
  614. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=254")
  615. assert response.status_code == 200
  616. mock_client.ams_load_filament.assert_called_once_with(254)
  617. assert "external" in response.json()["message"].lower()
  618. @pytest.mark.asyncio
  619. @pytest.mark.integration
  620. async def test_load_external_right_success(self, async_client: AsyncClient, printer_factory):
  621. """tray_id=255 → Ext-R on dual-nozzle H2D."""
  622. printer = await printer_factory(name="H2D")
  623. mock_client = MagicMock()
  624. mock_client.ams_load_filament.return_value = True
  625. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  626. mock_pm.get_client.return_value = mock_client
  627. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=255")
  628. assert response.status_code == 200
  629. mock_client.ams_load_filament.assert_called_once_with(255)
  630. assert "Ext-R" in response.json()["message"]
  631. @pytest.mark.asyncio
  632. @pytest.mark.integration
  633. async def test_load_mqtt_failure_returns_500(self, async_client: AsyncClient, printer_factory):
  634. printer = await printer_factory(name="P")
  635. mock_client = MagicMock()
  636. mock_client.ams_load_filament.return_value = False
  637. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  638. mock_pm.get_client.return_value = mock_client
  639. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=0")
  640. assert response.status_code == 500
  641. assert "failed" in response.json()["detail"].lower()
  642. # ── unload ───────────────────────────────────────────────────────────────
  643. @pytest.mark.asyncio
  644. @pytest.mark.integration
  645. async def test_unload_not_found(self, async_client: AsyncClient):
  646. response = await async_client.post("/api/v1/printers/99999/ams/unload")
  647. assert response.status_code == 404
  648. @pytest.mark.asyncio
  649. @pytest.mark.integration
  650. async def test_unload_not_connected(self, async_client: AsyncClient, printer_factory):
  651. printer = await printer_factory(name="Disconnected")
  652. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  653. mock_pm.get_client.return_value = None
  654. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/unload")
  655. assert response.status_code == 400
  656. assert "not connected" in response.json()["detail"].lower()
  657. @pytest.mark.asyncio
  658. @pytest.mark.integration
  659. async def test_unload_success(self, async_client: AsyncClient, printer_factory):
  660. printer = await printer_factory(name="P")
  661. mock_client = MagicMock()
  662. mock_client.ams_unload_filament.return_value = True
  663. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  664. mock_pm.get_client.return_value = mock_client
  665. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/unload")
  666. assert response.status_code == 200
  667. mock_client.ams_unload_filament.assert_called_once_with()
  668. assert response.json()["success"] is True
  669. @pytest.mark.asyncio
  670. @pytest.mark.integration
  671. async def test_unload_mqtt_failure_returns_500(self, async_client: AsyncClient, printer_factory):
  672. printer = await printer_factory(name="P")
  673. mock_client = MagicMock()
  674. mock_client.ams_unload_filament.return_value = False
  675. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  676. mock_pm.get_client.return_value = mock_client
  677. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/unload")
  678. assert response.status_code == 500
  679. assert "failed" in response.json()["detail"].lower()
  680. class TestConfigureAMSSlotAPI:
  681. """Integration tests for AMS slot configure endpoint — tray_info_idx resolution."""
  682. @pytest.mark.asyncio
  683. @pytest.mark.integration
  684. async def test_configure_not_connected(self, async_client: AsyncClient, printer_factory):
  685. """Verify error when printer is not connected."""
  686. printer = await printer_factory(name="Disconnected")
  687. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  688. mock_pm.get_client.return_value = None
  689. response = await async_client.post(
  690. f"/api/v1/printers/{printer.id}/slots/0/0/configure",
  691. params={
  692. "tray_info_idx": "GFL99",
  693. "tray_type": "PLA",
  694. "tray_sub_brands": "PLA Basic",
  695. "tray_color": "FF0000FF",
  696. "nozzle_temp_min": 190,
  697. "nozzle_temp_max": 230,
  698. },
  699. )
  700. assert response.status_code == 400
  701. assert "not connected" in response.json()["detail"].lower()
  702. @pytest.mark.asyncio
  703. @pytest.mark.integration
  704. async def test_configure_with_gf_id_keeps_it(self, async_client: AsyncClient, printer_factory):
  705. """Standard Bambu GF* filament IDs are sent as-is."""
  706. printer = await printer_factory(name="H2D")
  707. mock_client = MagicMock()
  708. mock_client.ams_set_filament_setting.return_value = True
  709. mock_client.extrusion_cali_sel.return_value = True
  710. mock_client.request_status_update.return_value = True
  711. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  712. mock_pm.get_client.return_value = mock_client
  713. mock_pm.get_status.return_value = None # No existing state
  714. response = await async_client.post(
  715. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  716. params={
  717. "tray_info_idx": "GFL05",
  718. "tray_type": "PLA",
  719. "tray_sub_brands": "PLA Basic",
  720. "tray_color": "FFFFFFFF",
  721. "nozzle_temp_min": 190,
  722. "nozzle_temp_max": 230,
  723. },
  724. )
  725. assert response.status_code == 200
  726. call_kwargs = mock_client.ams_set_filament_setting.call_args
  727. assert call_kwargs.kwargs["tray_info_idx"] == "GFL05"
  728. @pytest.mark.asyncio
  729. @pytest.mark.integration
  730. async def test_configure_pfus_sent_directly(self, async_client: AsyncClient, printer_factory):
  731. """PFUS* cloud-synced custom preset IDs are sent to the printer."""
  732. printer = await printer_factory(name="H2D")
  733. mock_client = MagicMock()
  734. mock_client.ams_set_filament_setting.return_value = True
  735. mock_client.extrusion_cali_sel.return_value = True
  736. mock_client.request_status_update.return_value = True
  737. mock_status = MagicMock()
  738. mock_status.raw_data = {"ams": {"ams": []}} # No existing tray data
  739. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  740. mock_pm.get_client.return_value = mock_client
  741. mock_pm.get_status.return_value = mock_status
  742. response = await async_client.post(
  743. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  744. params={
  745. "tray_info_idx": "PFUS9ac902733670a9",
  746. "tray_type": "PLA",
  747. "tray_sub_brands": "Devil Design PLA",
  748. "tray_color": "FF0000FF",
  749. "nozzle_temp_min": 190,
  750. "nozzle_temp_max": 230,
  751. },
  752. )
  753. assert response.status_code == 200
  754. call_kwargs = mock_client.ams_set_filament_setting.call_args
  755. assert call_kwargs.kwargs["tray_info_idx"] == "PFUS9ac902733670a9"
  756. @pytest.mark.asyncio
  757. @pytest.mark.integration
  758. async def test_configure_pfus_takes_priority_over_slot(self, async_client: AsyncClient, printer_factory):
  759. """Provided PFUS* preset takes priority over slot's existing preset."""
  760. printer = await printer_factory(name="H2D")
  761. mock_client = MagicMock()
  762. mock_client.ams_set_filament_setting.return_value = True
  763. mock_client.extrusion_cali_sel.return_value = True
  764. mock_client.request_status_update.return_value = True
  765. # Simulate slot already configured by slicer with cloud-synced preset
  766. mock_status = MagicMock()
  767. mock_status.raw_data = {
  768. "ams": {
  769. "ams": [
  770. {
  771. "id": 2,
  772. "tray": [
  773. {
  774. "id": 3,
  775. "tray_info_idx": "P4d64437",
  776. "tray_type": "PLA",
  777. "tray_color": "FF0000FF",
  778. }
  779. ],
  780. }
  781. ]
  782. }
  783. }
  784. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  785. mock_pm.get_client.return_value = mock_client
  786. mock_pm.get_status.return_value = mock_status
  787. response = await async_client.post(
  788. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  789. params={
  790. "tray_info_idx": "PFUS9ac902733670a9",
  791. "tray_type": "PLA",
  792. "tray_sub_brands": "Devil Design PLA",
  793. "tray_color": "FF0000FF",
  794. "nozzle_temp_min": 190,
  795. "nozzle_temp_max": 230,
  796. },
  797. )
  798. assert response.status_code == 200
  799. call_kwargs = mock_client.ams_set_filament_setting.call_args
  800. # Provided preset wins over slot's existing one
  801. assert call_kwargs.kwargs["tray_info_idx"] == "PFUS9ac902733670a9"
  802. @pytest.mark.asyncio
  803. @pytest.mark.integration
  804. async def test_configure_pfus_used_regardless_of_slot_material(self, async_client: AsyncClient, printer_factory):
  805. """Provided PFUS* preset is used even when slot has a different material."""
  806. printer = await printer_factory(name="H2D")
  807. mock_client = MagicMock()
  808. mock_client.ams_set_filament_setting.return_value = True
  809. mock_client.extrusion_cali_sel.return_value = True
  810. mock_client.request_status_update.return_value = True
  811. # Slot currently has PETG but user is configuring PLA
  812. mock_status = MagicMock()
  813. mock_status.raw_data = {
  814. "ams": {
  815. "ams": [
  816. {
  817. "id": 2,
  818. "tray": [{"id": 3, "tray_info_idx": "GFG99", "tray_type": "PETG", "tray_color": "FFFFFFFF"}],
  819. }
  820. ]
  821. }
  822. }
  823. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  824. mock_pm.get_client.return_value = mock_client
  825. mock_pm.get_status.return_value = mock_status
  826. response = await async_client.post(
  827. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  828. params={
  829. "tray_info_idx": "PFUS9ac902733670a9",
  830. "tray_type": "PLA",
  831. "tray_sub_brands": "Devil Design PLA",
  832. "tray_color": "FF0000FF",
  833. "nozzle_temp_min": 190,
  834. "nozzle_temp_max": 230,
  835. },
  836. )
  837. assert response.status_code == 200
  838. call_kwargs = mock_client.ams_set_filament_setting.call_args
  839. # Provided preset wins — slot's material is irrelevant
  840. assert call_kwargs.kwargs["tray_info_idx"] == "PFUS9ac902733670a9"
  841. @pytest.mark.asyncio
  842. @pytest.mark.integration
  843. async def test_configure_empty_id_uses_generic(self, async_client: AsyncClient, printer_factory):
  844. """Empty tray_info_idx (local preset) is replaced with generic."""
  845. printer = await printer_factory(name="H2D")
  846. mock_client = MagicMock()
  847. mock_client.ams_set_filament_setting.return_value = True
  848. mock_client.extrusion_cali_sel.return_value = True
  849. mock_client.request_status_update.return_value = True
  850. mock_status = MagicMock()
  851. mock_status.raw_data = {"ams": {"ams": []}}
  852. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  853. mock_pm.get_client.return_value = mock_client
  854. mock_pm.get_status.return_value = mock_status
  855. response = await async_client.post(
  856. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  857. params={
  858. "tray_info_idx": "",
  859. "tray_type": "PETG",
  860. "tray_sub_brands": "PETG Basic",
  861. "tray_color": "FFFFFFFF",
  862. "nozzle_temp_min": 220,
  863. "nozzle_temp_max": 260,
  864. },
  865. )
  866. assert response.status_code == 200
  867. call_kwargs = mock_client.ams_set_filament_setting.call_args
  868. assert call_kwargs.kwargs["tray_info_idx"] == "GFG99"
  869. @pytest.mark.asyncio
  870. @pytest.mark.integration
  871. async def test_configure_pfus_preserves_setting_id_pair(self, async_client: AsyncClient, printer_factory):
  872. """Both tray_info_idx=PFUS* and setting_id=PFUS* are forwarded untouched.
  873. Pins the end-to-end contract the frontend #1053 fix relies on: when the
  874. user configures a slot with a custom cloud preset whose cloud detail
  875. has filament_id=null, the frontend sends the setting_id in BOTH fields
  876. and the backend must not collapse either to a generic GF* ID.
  877. """
  878. printer = await printer_factory(name="H2D")
  879. mock_client = MagicMock()
  880. mock_client.ams_set_filament_setting.return_value = True
  881. mock_client.extrusion_cali_sel.return_value = True
  882. mock_client.request_status_update.return_value = True
  883. mock_status = MagicMock()
  884. mock_status.raw_data = {"ams": {"ams": []}}
  885. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  886. mock_pm.get_client.return_value = mock_client
  887. mock_pm.get_status.return_value = mock_status
  888. response = await async_client.post(
  889. f"/api/v1/printers/{printer.id}/slots/128/0/configure",
  890. params={
  891. "tray_info_idx": "PFUSa8fb76f9733e3c",
  892. "tray_type": "ABS",
  893. "tray_sub_brands": "Sting3D ABS",
  894. "tray_color": "000000FF",
  895. "nozzle_temp_min": 240,
  896. "nozzle_temp_max": 280,
  897. "setting_id": "PFUSa8fb76f9733e3c",
  898. },
  899. )
  900. assert response.status_code == 200
  901. call_kwargs = mock_client.ams_set_filament_setting.call_args
  902. assert call_kwargs.kwargs["tray_info_idx"] == "PFUSa8fb76f9733e3c"
  903. assert call_kwargs.kwargs["setting_id"] == "PFUSa8fb76f9733e3c"
  904. # Explicitly assert no generic-collapse happened for this HT slot.
  905. assert call_kwargs.kwargs["tray_info_idx"] != "GFB99"
  906. class TestSkipObjectsAPI:
  907. """Integration tests for skip objects endpoints."""
  908. # ========================================================================
  909. # Get printable objects endpoint
  910. # ========================================================================
  911. @pytest.mark.asyncio
  912. @pytest.mark.integration
  913. async def test_get_objects_not_found(self, async_client: AsyncClient):
  914. """Verify 404 for non-existent printer."""
  915. response = await async_client.get("/api/v1/printers/99999/print/objects")
  916. assert response.status_code == 404
  917. @pytest.mark.asyncio
  918. @pytest.mark.integration
  919. async def test_get_objects_not_connected(self, async_client: AsyncClient, printer_factory):
  920. """Verify error when printer is not connected."""
  921. printer = await printer_factory(name="Disconnected Printer")
  922. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  923. mock_pm.get_client.return_value = None
  924. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  925. assert response.status_code == 400
  926. assert "not connected" in response.json()["detail"].lower()
  927. @pytest.mark.asyncio
  928. @pytest.mark.integration
  929. async def test_get_objects_empty(self, async_client: AsyncClient, printer_factory):
  930. """Verify empty objects list when no print is active."""
  931. printer = await printer_factory(name="Idle Printer")
  932. mock_client = MagicMock()
  933. mock_client.state.printable_objects = {}
  934. mock_client.state.skipped_objects = []
  935. mock_client.state.state = "IDLE"
  936. mock_client.state.subtask_name = None # Prevent FTP download attempt
  937. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  938. mock_pm.get_client.return_value = mock_client
  939. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  940. assert response.status_code == 200
  941. result = response.json()
  942. assert result["objects"] == []
  943. assert result["total"] == 0
  944. assert result["skipped_count"] == 0
  945. assert result["is_printing"] is False
  946. @pytest.mark.asyncio
  947. @pytest.mark.integration
  948. async def test_get_objects_with_data(self, async_client: AsyncClient, printer_factory):
  949. """Verify objects list when print is active."""
  950. printer = await printer_factory(name="Printing Printer")
  951. mock_client = MagicMock()
  952. mock_client.state.printable_objects = {100: "Part A", 200: "Part B", 300: "Part C"}
  953. mock_client.state.skipped_objects = [200]
  954. mock_client.state.state = "RUNNING"
  955. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  956. mock_pm.get_client.return_value = mock_client
  957. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  958. assert response.status_code == 200
  959. result = response.json()
  960. assert result["total"] == 3
  961. assert result["skipped_count"] == 1
  962. assert result["is_printing"] is True
  963. # Check objects have correct structure
  964. objects_by_id = {obj["id"]: obj for obj in result["objects"]}
  965. assert objects_by_id[100]["name"] == "Part A"
  966. assert objects_by_id[100]["skipped"] is False
  967. assert objects_by_id[200]["name"] == "Part B"
  968. assert objects_by_id[200]["skipped"] is True
  969. assert objects_by_id[300]["name"] == "Part C"
  970. assert objects_by_id[300]["skipped"] is False
  971. # ========================================================================
  972. # Skip objects endpoint
  973. # ========================================================================
  974. @pytest.mark.asyncio
  975. @pytest.mark.integration
  976. async def test_get_objects_with_positions(self, async_client: AsyncClient, printer_factory):
  977. """Verify objects list includes position data when available."""
  978. printer = await printer_factory(name="Printing Printer")
  979. # New format with position data
  980. mock_client = MagicMock()
  981. mock_client.state.printable_objects = {
  982. 100: {"name": "Part A", "x": 50.0, "y": 100.0},
  983. 200: {"name": "Part B", "x": 150.0, "y": 100.0},
  984. }
  985. mock_client.state.skipped_objects = []
  986. mock_client.state.state = "RUNNING"
  987. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  988. mock_pm.get_client.return_value = mock_client
  989. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  990. assert response.status_code == 200
  991. result = response.json()
  992. assert result["total"] == 2
  993. # Check objects have position data
  994. objects_by_id = {obj["id"]: obj for obj in result["objects"]}
  995. assert objects_by_id[100]["name"] == "Part A"
  996. assert objects_by_id[100]["x"] == 50.0
  997. assert objects_by_id[100]["y"] == 100.0
  998. assert objects_by_id[200]["name"] == "Part B"
  999. assert objects_by_id[200]["x"] == 150.0
  1000. assert objects_by_id[200]["y"] == 100.0
  1001. # ========================================================================
  1002. # Skip objects endpoint
  1003. # ========================================================================
  1004. @pytest.mark.asyncio
  1005. @pytest.mark.integration
  1006. async def test_skip_objects_not_found(self, async_client: AsyncClient):
  1007. """Verify 404 for non-existent printer."""
  1008. response = await async_client.post("/api/v1/printers/99999/print/skip-objects", json=[100])
  1009. assert response.status_code == 404
  1010. @pytest.mark.asyncio
  1011. @pytest.mark.integration
  1012. async def test_skip_objects_not_connected(self, async_client: AsyncClient, printer_factory):
  1013. """Verify error when printer is not connected."""
  1014. printer = await printer_factory(name="Disconnected Printer")
  1015. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1016. mock_pm.get_client.return_value = None
  1017. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[100])
  1018. assert response.status_code == 400
  1019. assert "not connected" in response.json()["detail"].lower()
  1020. @pytest.mark.asyncio
  1021. @pytest.mark.integration
  1022. async def test_skip_objects_empty_list(self, async_client: AsyncClient, printer_factory):
  1023. """Verify error when no object IDs provided."""
  1024. printer = await printer_factory(name="Printing Printer")
  1025. mock_client = MagicMock()
  1026. mock_client.state.printable_objects = {100: "Part A"}
  1027. mock_client.state.skipped_objects = []
  1028. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1029. mock_pm.get_client.return_value = mock_client
  1030. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[])
  1031. assert response.status_code == 400
  1032. assert "no object" in response.json()["detail"].lower()
  1033. @pytest.mark.asyncio
  1034. @pytest.mark.integration
  1035. async def test_skip_objects_invalid_id(self, async_client: AsyncClient, printer_factory):
  1036. """Verify error when object ID doesn't exist."""
  1037. printer = await printer_factory(name="Printing Printer")
  1038. mock_client = MagicMock()
  1039. mock_client.state.printable_objects = {100: "Part A"}
  1040. mock_client.state.skipped_objects = []
  1041. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1042. mock_pm.get_client.return_value = mock_client
  1043. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[999])
  1044. assert response.status_code == 400
  1045. assert "invalid" in response.json()["detail"].lower()
  1046. @pytest.mark.asyncio
  1047. @pytest.mark.integration
  1048. async def test_skip_objects_success(self, async_client: AsyncClient, printer_factory):
  1049. """Verify successful skip objects request."""
  1050. printer = await printer_factory(name="Printing Printer")
  1051. mock_client = MagicMock()
  1052. mock_client.state.printable_objects = {100: "Part A", 200: "Part B"}
  1053. mock_client.state.skipped_objects = []
  1054. mock_client.skip_objects.return_value = True
  1055. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1056. mock_pm.get_client.return_value = mock_client
  1057. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[100])
  1058. assert response.status_code == 200
  1059. result = response.json()
  1060. assert result["success"] is True
  1061. assert 100 in result["skipped_objects"]
  1062. mock_client.skip_objects.assert_called_once_with([100])
  1063. @pytest.mark.asyncio
  1064. @pytest.mark.integration
  1065. async def test_skip_objects_multiple(self, async_client: AsyncClient, printer_factory):
  1066. """Verify skipping multiple objects at once."""
  1067. printer = await printer_factory(name="Printing Printer")
  1068. mock_client = MagicMock()
  1069. mock_client.state.printable_objects = {100: "Part A", 200: "Part B", 300: "Part C"}
  1070. mock_client.state.skipped_objects = []
  1071. mock_client.skip_objects.return_value = True
  1072. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1073. mock_pm.get_client.return_value = mock_client
  1074. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[100, 200])
  1075. assert response.status_code == 200
  1076. result = response.json()
  1077. assert result["success"] is True
  1078. assert 100 in result["skipped_objects"]
  1079. assert 200 in result["skipped_objects"]
  1080. mock_client.skip_objects.assert_called_once_with([100, 200])
  1081. class TestChamberLightAPI:
  1082. """Integration tests for chamber light control endpoint."""
  1083. @pytest.mark.asyncio
  1084. @pytest.mark.integration
  1085. async def test_chamber_light_not_found(self, async_client: AsyncClient):
  1086. """Verify 404 for non-existent printer."""
  1087. response = await async_client.post("/api/v1/printers/99999/chamber-light?on=true")
  1088. assert response.status_code == 404
  1089. @pytest.mark.asyncio
  1090. @pytest.mark.integration
  1091. async def test_chamber_light_not_connected(self, async_client: AsyncClient, printer_factory):
  1092. """Verify error when printer is not connected."""
  1093. printer = await printer_factory(name="Disconnected Printer")
  1094. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1095. mock_pm.get_client.return_value = None
  1096. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=true")
  1097. assert response.status_code == 400
  1098. assert "not connected" in response.json()["detail"].lower()
  1099. @pytest.mark.asyncio
  1100. @pytest.mark.integration
  1101. async def test_chamber_light_on_success(self, async_client: AsyncClient, printer_factory):
  1102. """Verify successful chamber light on request."""
  1103. printer = await printer_factory(name="Test Printer")
  1104. mock_client = MagicMock()
  1105. mock_client.set_chamber_light.return_value = True
  1106. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1107. mock_pm.get_client.return_value = mock_client
  1108. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=true")
  1109. assert response.status_code == 200
  1110. result = response.json()
  1111. assert result["success"] is True
  1112. assert "on" in result["message"].lower()
  1113. mock_client.set_chamber_light.assert_called_once_with(True)
  1114. @pytest.mark.asyncio
  1115. @pytest.mark.integration
  1116. async def test_chamber_light_off_success(self, async_client: AsyncClient, printer_factory):
  1117. """Verify successful chamber light off request."""
  1118. printer = await printer_factory(name="Test Printer")
  1119. mock_client = MagicMock()
  1120. mock_client.set_chamber_light.return_value = True
  1121. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1122. mock_pm.get_client.return_value = mock_client
  1123. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=false")
  1124. assert response.status_code == 200
  1125. result = response.json()
  1126. assert result["success"] is True
  1127. assert "off" in result["message"].lower()
  1128. mock_client.set_chamber_light.assert_called_once_with(False)
  1129. @pytest.mark.asyncio
  1130. @pytest.mark.integration
  1131. async def test_chamber_light_failure(self, async_client: AsyncClient, printer_factory):
  1132. """Verify error handling when chamber light control fails."""
  1133. printer = await printer_factory(name="Test Printer")
  1134. mock_client = MagicMock()
  1135. mock_client.set_chamber_light.return_value = False
  1136. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1137. mock_pm.get_client.return_value = mock_client
  1138. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=true")
  1139. assert response.status_code == 500
  1140. assert "failed" in response.json()["detail"].lower()
  1141. class TestAirductModeAPI:
  1142. """Integration tests for the airduct mode endpoint (P2S/H2*)."""
  1143. @pytest.mark.asyncio
  1144. @pytest.mark.integration
  1145. async def test_invalid_mode_rejected(self, async_client: AsyncClient, printer_factory):
  1146. printer = await printer_factory(name="P", model="P2S")
  1147. response = await async_client.post(f"/api/v1/printers/{printer.id}/airduct-mode?mode=foo")
  1148. assert response.status_code == 400
  1149. @pytest.mark.asyncio
  1150. @pytest.mark.integration
  1151. async def test_not_connected(self, async_client: AsyncClient, printer_factory):
  1152. printer = await printer_factory(name="P", model="P2S")
  1153. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1154. mock_pm.get_client.return_value = None
  1155. response = await async_client.post(f"/api/v1/printers/{printer.id}/airduct-mode?mode=cooling")
  1156. assert response.status_code == 400
  1157. @pytest.mark.asyncio
  1158. @pytest.mark.integration
  1159. async def test_cooling_success(self, async_client: AsyncClient, printer_factory):
  1160. printer = await printer_factory(name="P", model="P2S")
  1161. mock_client = MagicMock()
  1162. mock_client.set_airduct_mode.return_value = True
  1163. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1164. mock_pm.get_client.return_value = mock_client
  1165. response = await async_client.post(f"/api/v1/printers/{printer.id}/airduct-mode?mode=cooling")
  1166. assert response.status_code == 200
  1167. assert response.json()["success"] is True
  1168. mock_client.set_airduct_mode.assert_called_once_with("cooling")
  1169. @pytest.mark.asyncio
  1170. @pytest.mark.integration
  1171. async def test_heating_failure_returns_500(self, async_client: AsyncClient, printer_factory):
  1172. printer = await printer_factory(name="P", model="P2S")
  1173. mock_client = MagicMock()
  1174. mock_client.set_airduct_mode.return_value = False
  1175. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1176. mock_pm.get_client.return_value = mock_client
  1177. response = await async_client.post(f"/api/v1/printers/{printer.id}/airduct-mode?mode=heating")
  1178. assert response.status_code == 500
  1179. class TestClearHMSErrorsAPI:
  1180. """Integration tests for clear HMS errors endpoint."""
  1181. @pytest.mark.asyncio
  1182. @pytest.mark.integration
  1183. async def test_clear_hms_errors_not_found(self, async_client: AsyncClient):
  1184. """Verify 404 for non-existent printer."""
  1185. response = await async_client.post("/api/v1/printers/99999/hms/clear")
  1186. assert response.status_code == 404
  1187. @pytest.mark.asyncio
  1188. @pytest.mark.integration
  1189. async def test_clear_hms_errors_not_connected(self, async_client: AsyncClient, printer_factory):
  1190. """Verify error when printer is not connected."""
  1191. printer = await printer_factory(name="Disconnected Printer")
  1192. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1193. mock_pm.get_client.return_value = None
  1194. response = await async_client.post(f"/api/v1/printers/{printer.id}/hms/clear")
  1195. assert response.status_code == 400
  1196. assert "not connected" in response.json()["detail"].lower()
  1197. @pytest.mark.asyncio
  1198. @pytest.mark.integration
  1199. async def test_clear_hms_errors_success(self, async_client: AsyncClient, printer_factory):
  1200. """Verify successful clear HMS errors request."""
  1201. printer = await printer_factory(name="Test Printer")
  1202. mock_client = MagicMock()
  1203. mock_client.clear_hms_errors.return_value = True
  1204. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1205. mock_pm.get_client.return_value = mock_client
  1206. response = await async_client.post(f"/api/v1/printers/{printer.id}/hms/clear")
  1207. assert response.status_code == 200
  1208. result = response.json()
  1209. assert result["success"] is True
  1210. assert "cleared" in result["message"].lower()
  1211. mock_client.clear_hms_errors.assert_called_once()
  1212. @pytest.mark.asyncio
  1213. @pytest.mark.integration
  1214. async def test_clear_hms_errors_failure(self, async_client: AsyncClient, printer_factory):
  1215. """Verify error handling when clear HMS errors fails."""
  1216. printer = await printer_factory(name="Test Printer")
  1217. mock_client = MagicMock()
  1218. mock_client.clear_hms_errors.return_value = False
  1219. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1220. mock_pm.get_client.return_value = mock_client
  1221. response = await async_client.post(f"/api/v1/printers/{printer.id}/hms/clear")
  1222. assert response.status_code == 500
  1223. assert "failed" in response.json()["detail"].lower()