test_printers_api.py 121 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999
  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 AsyncMock, MagicMock, patch
  5. from urllib.parse import unquote
  6. import pytest
  7. from httpx import AsyncClient
  8. from sqlalchemy import select
  9. @pytest.fixture(autouse=True)
  10. def _mock_printer_test_connection():
  11. """Default mock: connection test returns success.
  12. POST /printers/ now refuses to persist a printer when the MQTT
  13. connection probe fails (would otherwise leave an empty card in the
  14. dashboard for a mistyped access code). Existing tests assume the
  15. save succeeds, so we mock the probe green by default; the failure
  16. branch is exercised by a dedicated test below.
  17. """
  18. with patch(
  19. "backend.app.services.printer_manager.printer_manager.test_connection",
  20. new=AsyncMock(return_value={"success": True, "state": "IDLE", "model": "X1C"}),
  21. ) as m:
  22. yield m
  23. class TestPrintersAPI:
  24. """Integration tests for /api/v1/printers/ endpoints."""
  25. # ========================================================================
  26. # List endpoints
  27. # ========================================================================
  28. @pytest.mark.asyncio
  29. @pytest.mark.integration
  30. async def test_list_printers_empty(self, async_client: AsyncClient):
  31. """Verify empty list is returned when no printers exist."""
  32. response = await async_client.get("/api/v1/printers/")
  33. assert response.status_code == 200
  34. assert response.json() == []
  35. @pytest.mark.asyncio
  36. @pytest.mark.integration
  37. async def test_list_printers_with_data(self, async_client: AsyncClient, printer_factory, db_session):
  38. """Verify list returns existing printers."""
  39. await printer_factory(name="Test Printer")
  40. response = await async_client.get("/api/v1/printers/")
  41. assert response.status_code == 200
  42. data = response.json()
  43. assert len(data) >= 1
  44. assert any(p["name"] == "Test Printer" for p in data)
  45. # ========================================================================
  46. # Create endpoints
  47. # ========================================================================
  48. @pytest.mark.asyncio
  49. @pytest.mark.integration
  50. async def test_create_printer(self, async_client: AsyncClient):
  51. """Verify printer can be created."""
  52. data = {
  53. "name": "New Printer",
  54. "serial_number": "00M09A111111111",
  55. "ip_address": "192.168.1.100",
  56. "access_code": "12345678",
  57. "is_active": True,
  58. "model": "X1C",
  59. }
  60. response = await async_client.post("/api/v1/printers/", json=data)
  61. assert response.status_code == 200
  62. result = response.json()
  63. assert result["name"] == "New Printer"
  64. assert result["serial_number"] == "00M09A111111111"
  65. assert result["model"] == "X1C"
  66. @pytest.mark.asyncio
  67. @pytest.mark.integration
  68. async def test_create_printer_with_hostname(self, async_client: AsyncClient):
  69. """Verify printer can be created with a hostname instead of IP address."""
  70. data = {
  71. "name": "DNS Printer",
  72. "serial_number": "00M09A555555555",
  73. "ip_address": "printer.local",
  74. "access_code": "12345678",
  75. "model": "P1S",
  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["name"] == "DNS Printer"
  81. assert result["ip_address"] == "printer.local"
  82. @pytest.mark.asyncio
  83. @pytest.mark.integration
  84. async def test_create_printer_with_fqdn(self, async_client: AsyncClient):
  85. """Verify printer can be created with a fully qualified domain name."""
  86. data = {
  87. "name": "FQDN Printer",
  88. "serial_number": "00M09A666666666",
  89. "ip_address": "my-printer.home.lan",
  90. "access_code": "12345678",
  91. "model": "X1C",
  92. }
  93. response = await async_client.post("/api/v1/printers/", json=data)
  94. assert response.status_code == 200
  95. result = response.json()
  96. assert result["ip_address"] == "my-printer.home.lan"
  97. @pytest.mark.asyncio
  98. @pytest.mark.integration
  99. async def test_create_printer_invalid_hostname(self, async_client: AsyncClient):
  100. """Verify invalid hostnames are rejected."""
  101. data = {
  102. "name": "Bad Printer",
  103. "serial_number": "00M09A777777777",
  104. "ip_address": "-invalid",
  105. "access_code": "12345678",
  106. }
  107. response = await async_client.post("/api/v1/printers/", json=data)
  108. assert response.status_code == 422
  109. @pytest.mark.asyncio
  110. @pytest.mark.integration
  111. async def test_create_printer_duplicate_serial(self, async_client: AsyncClient, printer_factory, db_session):
  112. """Verify duplicate serial number is rejected."""
  113. await printer_factory(serial_number="00M09A222222222")
  114. data = {
  115. "name": "Duplicate Printer",
  116. "serial_number": "00M09A222222222",
  117. "ip_address": "192.168.1.101",
  118. "access_code": "12345678",
  119. }
  120. response = await async_client.post("/api/v1/printers/", json=data)
  121. # Should fail due to duplicate serial
  122. assert response.status_code in [400, 409, 422, 500]
  123. @pytest.mark.asyncio
  124. @pytest.mark.integration
  125. async def test_create_printer_rejects_when_mqtt_probe_fails(self, async_client: AsyncClient, db_session):
  126. """Wrong access code / unreachable IP must NOT persist the printer.
  127. Regression: users were reporting empty / never-connecting printer
  128. cards that traced back to a mistyped access code. The create route
  129. now runs an MQTT probe up front and returns 400 if it fails — the
  130. row is never written.
  131. """
  132. data = {
  133. "name": "Bad Code Printer",
  134. "serial_number": "00M09A999999999",
  135. "ip_address": "192.168.1.250",
  136. "access_code": "WRONG-CODE",
  137. "is_active": True,
  138. "model": "X1C",
  139. }
  140. with patch(
  141. "backend.app.services.printer_manager.printer_manager.test_connection",
  142. new=AsyncMock(return_value={"success": False, "state": None, "model": None}),
  143. ):
  144. response = await async_client.post("/api/v1/printers/", json=data)
  145. assert response.status_code == 400
  146. detail = response.json()["detail"]
  147. # Backend returns a stable code for the frontend i18n layer to map;
  148. # the message field is an English fallback for non-UI clients.
  149. assert detail["code"] == "printer_connection_failed"
  150. assert "connect" in detail["message"].lower()
  151. # And critically: the printer row was never persisted.
  152. from backend.app.models.printer import Printer
  153. result = await db_session.execute(select(Printer).where(Printer.serial_number == "00M09A999999999"))
  154. assert result.scalar_one_or_none() is None, "Failed-probe printer must not be persisted"
  155. # ========================================================================
  156. # Get single endpoint
  157. # ========================================================================
  158. @pytest.mark.asyncio
  159. @pytest.mark.integration
  160. async def test_get_printer(self, async_client: AsyncClient, printer_factory, db_session):
  161. """Verify single printer can be retrieved."""
  162. printer = await printer_factory(name="Get Test Printer")
  163. response = await async_client.get(f"/api/v1/printers/{printer.id}")
  164. assert response.status_code == 200
  165. result = response.json()
  166. assert result["id"] == printer.id
  167. assert result["name"] == "Get Test Printer"
  168. @pytest.mark.asyncio
  169. @pytest.mark.integration
  170. async def test_get_printer_not_found(self, async_client: AsyncClient):
  171. """Verify 404 for non-existent printer."""
  172. response = await async_client.get("/api/v1/printers/9999")
  173. assert response.status_code == 404
  174. # ========================================================================
  175. # Update endpoints
  176. # ========================================================================
  177. @pytest.mark.asyncio
  178. @pytest.mark.integration
  179. async def test_update_printer_name(self, async_client: AsyncClient, printer_factory, db_session):
  180. """Verify printer name can be updated."""
  181. printer = await printer_factory(name="Original Name")
  182. response = await async_client.patch(f"/api/v1/printers/{printer.id}", json={"name": "Updated Name"})
  183. assert response.status_code == 200
  184. assert response.json()["name"] == "Updated Name"
  185. @pytest.mark.asyncio
  186. @pytest.mark.integration
  187. async def test_update_printer_active_status(self, async_client: AsyncClient, printer_factory, db_session):
  188. """Verify printer active status can be updated."""
  189. printer = await printer_factory(is_active=True)
  190. response = await async_client.patch(f"/api/v1/printers/{printer.id}", json={"is_active": False})
  191. assert response.status_code == 200
  192. assert response.json()["is_active"] is False
  193. @pytest.mark.asyncio
  194. @pytest.mark.integration
  195. async def test_update_printer_auto_archive(self, async_client: AsyncClient, printer_factory, db_session):
  196. """Verify auto_archive setting can be updated."""
  197. printer = await printer_factory(auto_archive=True)
  198. response = await async_client.patch(f"/api/v1/printers/{printer.id}", json={"auto_archive": False})
  199. assert response.status_code == 200
  200. assert response.json()["auto_archive"] is False
  201. @pytest.mark.asyncio
  202. @pytest.mark.integration
  203. async def test_update_nonexistent_printer(self, async_client: AsyncClient):
  204. """Verify updating non-existent printer returns 404."""
  205. response = await async_client.patch("/api/v1/printers/9999", json={"name": "New Name"})
  206. assert response.status_code == 404
  207. # ========================================================================
  208. # Delete endpoints
  209. # ========================================================================
  210. @pytest.mark.asyncio
  211. @pytest.mark.integration
  212. async def test_delete_printer(self, async_client: AsyncClient, printer_factory, db_session):
  213. """Verify printer can be deleted."""
  214. printer = await printer_factory()
  215. printer_id = printer.id
  216. response = await async_client.delete(f"/api/v1/printers/{printer_id}")
  217. assert response.status_code == 200
  218. # Verify deleted
  219. response = await async_client.get(f"/api/v1/printers/{printer_id}")
  220. assert response.status_code == 404
  221. @pytest.mark.asyncio
  222. @pytest.mark.integration
  223. async def test_delete_nonexistent_printer(self, async_client: AsyncClient):
  224. """Verify deleting non-existent printer returns 404."""
  225. response = await async_client.delete("/api/v1/printers/9999")
  226. assert response.status_code == 404
  227. # ========================================================================
  228. # File download endpoint — non-ASCII filename regression (#1245)
  229. # ========================================================================
  230. @pytest.mark.asyncio
  231. @pytest.mark.integration
  232. @pytest.mark.parametrize(
  233. ("filename", "ascii_fallback"),
  234. [
  235. ("龙泡泡石墩子_p2s_ok.gcode.3mf", "p2s_ok.gcode.3mf"),
  236. ("こんにちは.gcode.3mf", "gcode.3mf"),
  237. ("résumé.gcode.3mf", "rsum.gcode.3mf"),
  238. ("مرحبا.gcode.3mf", "gcode.3mf"),
  239. ("文件.3mf", "3mf"),
  240. ("hello.3mf", "hello.3mf"),
  241. ],
  242. )
  243. async def test_download_printer_file_non_ascii_filename(
  244. self,
  245. async_client: AsyncClient,
  246. printer_factory,
  247. filename: str,
  248. ascii_fallback: str,
  249. db_session,
  250. ):
  251. """Non-ASCII filenames must not crash header encoding (issue #1245)."""
  252. printer = await printer_factory()
  253. file_bytes = b"fake 3mf content"
  254. with patch(
  255. "backend.app.api.routes.printers.download_file_bytes_async",
  256. new=AsyncMock(return_value=file_bytes),
  257. ):
  258. response = await async_client.get(
  259. f"/api/v1/printers/{printer.id}/files/download",
  260. params={"path": f"/cache/{filename}"},
  261. )
  262. assert response.status_code == 200
  263. assert response.content == file_bytes
  264. content_disposition = response.headers["content-disposition"]
  265. assert f'filename="{ascii_fallback}"' in content_disposition
  266. assert "filename*=UTF-8''" in content_disposition
  267. encoded_name = content_disposition.split("filename*=UTF-8''", 1)[1]
  268. assert unquote(encoded_name) == filename
  269. # ========================================================================
  270. # Status endpoint
  271. # ========================================================================
  272. @pytest.mark.asyncio
  273. @pytest.mark.integration
  274. async def test_get_printer_status(
  275. self, async_client: AsyncClient, printer_factory, mock_printer_manager, db_session
  276. ):
  277. """Verify printer status can be retrieved."""
  278. printer = await printer_factory()
  279. response = await async_client.get(f"/api/v1/printers/{printer.id}/status")
  280. assert response.status_code == 200
  281. result = response.json()
  282. assert "connected" in result
  283. assert "state" in result
  284. @pytest.mark.asyncio
  285. @pytest.mark.integration
  286. async def test_get_printer_status_not_found(self, async_client: AsyncClient):
  287. """Verify 404 for status of non-existent printer."""
  288. response = await async_client.get("/api/v1/printers/9999/status")
  289. assert response.status_code == 404
  290. @pytest.mark.asyncio
  291. @pytest.mark.integration
  292. async def test_get_printer_status_includes_fila_switch_when_installed(
  293. self, async_client: AsyncClient, printer_factory, db_session
  294. ):
  295. """When the FTS accessory is installed, the status response must include
  296. the fila_switch object with the routing arrays. See #1162.
  297. The accessory is detected from print.device.fila_switch in MQTT;
  298. we feed a PrinterState with FilaSwitchState(installed=True, ...) and
  299. confirm it survives the schema serialization round-trip.
  300. """
  301. from unittest.mock import MagicMock, patch
  302. from backend.app.services.bambu_mqtt import FilaSwitchState, PrinterState
  303. printer = await printer_factory()
  304. state = PrinterState()
  305. state.connected = True
  306. state.state = "IDLE"
  307. state.fila_switch = FilaSwitchState(
  308. installed=True,
  309. in_slots=[-1, 2],
  310. out_extruders=[0, 1],
  311. stat=0,
  312. info=2,
  313. )
  314. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  315. mock_pm.get_status = MagicMock(return_value=state)
  316. mock_pm.is_awaiting_plate_clear = MagicMock(return_value=False)
  317. response = await async_client.get(f"/api/v1/printers/{printer.id}/status")
  318. assert response.status_code == 200
  319. result = response.json()
  320. assert result["fila_switch"] is not None
  321. assert result["fila_switch"]["installed"] is True
  322. assert result["fila_switch"]["in_slots"] == [-1, 2]
  323. assert result["fila_switch"]["out_extruders"] == [0, 1]
  324. assert result["fila_switch"]["stat"] == 0
  325. assert result["fila_switch"]["info"] == 2
  326. @pytest.mark.asyncio
  327. @pytest.mark.integration
  328. async def test_cover_uses_dispatched_plate_when_gcode_file_lacks_path(
  329. self, async_client: AsyncClient, printer_factory, db_session, tmp_path
  330. ):
  331. """When firmware drops the plate path from gcode_file (e.g. P1S
  332. 01.10.00.00, #1166), the dispatched-plate record must take precedence
  333. and serve plate 4's thumbnail instead of falling back to plate_1.png."""
  334. import io
  335. import zipfile
  336. from unittest.mock import MagicMock, patch
  337. from backend.app.services.bambu_ftp import cache_3mf_download
  338. from backend.app.services.bambu_mqtt import PrinterState
  339. printer = await printer_factory()
  340. # Build a 3MF that mimics a "true" multi-plate archive: thumbnails
  341. # for plates 1..4 are all present, gcode files for plates 1..4 are
  342. # all present. Without the dispatch record we'd default to plate_1.png.
  343. threemf_path = tmp_path / "MyModel.3mf"
  344. with zipfile.ZipFile(threemf_path, "w") as zf:
  345. for plate in range(1, 5):
  346. zf.writestr(f"Metadata/plate_{plate}.png", f"PLATE_{plate}_PNG".encode())
  347. zf.writestr(f"Metadata/plate_{plate}.gcode", f"; plate {plate} gcode\n")
  348. cache_3mf_download(printer.id, "MyModel.3mf", threemf_path)
  349. state = PrinterState()
  350. state.connected = True
  351. state.state = "RUNNING"
  352. state.subtask_name = "MyModel"
  353. state.gcode_file = "MyModel.3mf" # firmware drops plate path
  354. state.dispatched_plate_id = 4
  355. state.dispatched_subtask = "MyModel"
  356. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  357. mock_pm.get_status = MagicMock(return_value=state)
  358. mock_pm.is_awaiting_plate_clear = MagicMock(return_value=False)
  359. response = await async_client.get(f"/api/v1/printers/{printer.id}/cover")
  360. assert response.status_code == 200
  361. assert response.content == b"PLATE_4_PNG"
  362. @pytest.mark.asyncio
  363. @pytest.mark.integration
  364. async def test_cover_3mf_scan_fallback_for_per_plate_archive(
  365. self, async_client: AsyncClient, printer_factory, db_session, tmp_path
  366. ):
  367. """Per-plate archives sliced separately in Bambu Studio contain a
  368. single Metadata/plate_N.gcode (the active plate) but bundle thumbnails
  369. for every plate. With no dispatch record (e.g. dispatched via Studio
  370. directly) and no plate path in gcode_file, the route must scan the
  371. 3MF and pick plate N's thumbnail. See #1166 option 4."""
  372. import zipfile
  373. from unittest.mock import MagicMock, patch
  374. from backend.app.services.bambu_ftp import cache_3mf_download
  375. from backend.app.services.bambu_mqtt import PrinterState
  376. printer = await printer_factory()
  377. # Per-plate archive: thumbnails for all plates, gcode for plate 3 only.
  378. threemf_path = tmp_path / "PerPlate.3mf"
  379. with zipfile.ZipFile(threemf_path, "w") as zf:
  380. for plate in range(1, 5):
  381. zf.writestr(f"Metadata/plate_{plate}.png", f"PLATE_{plate}_PNG".encode())
  382. zf.writestr("Metadata/plate_3.gcode", "; only plate 3 has gcode\n")
  383. cache_3mf_download(printer.id, "PerPlate.3mf", threemf_path)
  384. state = PrinterState()
  385. state.connected = True
  386. state.state = "RUNNING"
  387. state.subtask_name = "PerPlate"
  388. state.gcode_file = "PerPlate.3mf"
  389. # No dispatch record (Studio-direct dispatch).
  390. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  391. mock_pm.get_status = MagicMock(return_value=state)
  392. mock_pm.is_awaiting_plate_clear = MagicMock(return_value=False)
  393. response = await async_client.get(f"/api/v1/printers/{printer.id}/cover")
  394. assert response.status_code == 200
  395. assert response.content == b"PLATE_3_PNG"
  396. @pytest.mark.asyncio
  397. @pytest.mark.integration
  398. async def test_cover_negative_cache_skips_repeat_ftp_fanout(
  399. self, async_client: AsyncClient, printer_factory, db_session
  400. ):
  401. """#1420: when every FTP path returns 550 for the current subtask, the
  402. next request for the same subtask must short-circuit to 404 instead of
  403. replaying the 8-path FTP fan-out (which starves the printer's single
  404. FTP socket and flooded the user's logs)."""
  405. from unittest.mock import AsyncMock, MagicMock, patch
  406. from backend.app.api.routes.printers import _cover_404_cache, _cover_cache
  407. from backend.app.services.bambu_mqtt import PrinterState
  408. printer = await printer_factory()
  409. _cover_cache.pop(printer.id, None)
  410. _cover_404_cache.pop(printer.id, None)
  411. state = PrinterState()
  412. state.connected = True
  413. state.state = "RUNNING"
  414. state.subtask_name = "OrphanPrint"
  415. state.gcode_file = "OrphanPrint.3mf"
  416. ftp_mock = AsyncMock(return_value=False)
  417. with (
  418. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  419. patch("backend.app.api.routes.printers.download_file_try_paths_async", ftp_mock),
  420. ):
  421. mock_pm.get_status = MagicMock(return_value=state)
  422. mock_pm.is_awaiting_plate_clear = MagicMock(return_value=False)
  423. r1 = await async_client.get(f"/api/v1/printers/{printer.id}/cover")
  424. r2 = await async_client.get(f"/api/v1/printers/{printer.id}/cover")
  425. assert r1.status_code == 404
  426. assert r2.status_code == 404
  427. # First call retries internally; second call must short-circuit before FTP.
  428. first_call_count = ftp_mock.await_count
  429. assert first_call_count >= 1
  430. # Second request didn't add to the count: the negative cache held.
  431. assert ftp_mock.await_count == first_call_count
  432. @pytest.mark.asyncio
  433. @pytest.mark.integration
  434. async def test_get_printer_status_omits_fila_switch_when_not_installed(
  435. self, async_client: AsyncClient, printer_factory, db_session
  436. ):
  437. """Without the FTS accessory, fila_switch must be null so the frontend
  438. keeps applying the per-extruder filter on regular dual-nozzle printers."""
  439. from unittest.mock import MagicMock, patch
  440. from backend.app.services.bambu_mqtt import PrinterState
  441. printer = await printer_factory()
  442. state = PrinterState()
  443. state.connected = True
  444. state.state = "IDLE"
  445. # default fila_switch — installed = False
  446. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  447. mock_pm.get_status = MagicMock(return_value=state)
  448. mock_pm.is_awaiting_plate_clear = MagicMock(return_value=False)
  449. response = await async_client.get(f"/api/v1/printers/{printer.id}/status")
  450. assert response.status_code == 200
  451. result = response.json()
  452. assert result["fila_switch"] is None
  453. # ========================================================================
  454. # Test connection endpoint
  455. # ========================================================================
  456. class TestPrinterDataIntegrity:
  457. """Tests for printer data integrity."""
  458. @pytest.mark.asyncio
  459. @pytest.mark.integration
  460. async def test_printer_stores_all_fields(self, async_client: AsyncClient, printer_factory, db_session):
  461. """Verify printer stores all fields correctly."""
  462. printer = await printer_factory(
  463. name="Full Test Printer",
  464. serial_number="00M09A444444444",
  465. ip_address="192.168.1.150",
  466. model="P1S",
  467. is_active=True,
  468. auto_archive=False,
  469. )
  470. response = await async_client.get(f"/api/v1/printers/{printer.id}")
  471. assert response.status_code == 200
  472. result = response.json()
  473. assert result["name"] == "Full Test Printer"
  474. assert result["serial_number"] == "00M09A444444444"
  475. assert result["ip_address"] == "192.168.1.150"
  476. assert result["model"] == "P1S"
  477. assert result["is_active"] is True
  478. assert result["auto_archive"] is False
  479. @pytest.mark.asyncio
  480. @pytest.mark.integration
  481. async def test_printer_update_persists(self, async_client: AsyncClient, printer_factory, db_session):
  482. """CRITICAL: Verify printer updates persist."""
  483. printer = await printer_factory(name="Original", is_active=True)
  484. # Update
  485. await async_client.patch(f"/api/v1/printers/{printer.id}", json={"name": "Updated", "is_active": False})
  486. # Verify persistence
  487. response = await async_client.get(f"/api/v1/printers/{printer.id}")
  488. result = response.json()
  489. assert result["name"] == "Updated"
  490. assert result["is_active"] is False
  491. # ========================================================================
  492. # Refresh status endpoint
  493. # ========================================================================
  494. @pytest.mark.asyncio
  495. @pytest.mark.integration
  496. async def test_refresh_status_not_found(self, async_client: AsyncClient):
  497. """Verify 404 for non-existent printer."""
  498. response = await async_client.post("/api/v1/printers/99999/refresh-status")
  499. assert response.status_code == 404
  500. @pytest.mark.asyncio
  501. @pytest.mark.integration
  502. async def test_refresh_status_not_connected(self, async_client: AsyncClient, printer_factory):
  503. """Verify 400 when printer is not connected."""
  504. printer = await printer_factory(name="Disconnected Printer")
  505. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  506. mock_pm.request_status_update.return_value = False
  507. response = await async_client.post(f"/api/v1/printers/{printer.id}/refresh-status")
  508. assert response.status_code == 400
  509. assert "not connected" in response.json()["detail"].lower()
  510. @pytest.mark.asyncio
  511. @pytest.mark.integration
  512. async def test_refresh_status_success(self, async_client: AsyncClient, printer_factory):
  513. """Verify successful refresh request."""
  514. printer = await printer_factory(name="Connected Printer")
  515. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  516. mock_pm.request_status_update.return_value = True
  517. response = await async_client.post(f"/api/v1/printers/{printer.id}/refresh-status")
  518. assert response.status_code == 200
  519. assert response.json()["status"] == "refresh_requested"
  520. mock_pm.request_status_update.assert_called_once_with(printer.id)
  521. # ========================================================================
  522. # Current print user endpoint (Issue #206)
  523. # ========================================================================
  524. @pytest.mark.asyncio
  525. @pytest.mark.integration
  526. async def test_get_current_print_user_not_found(self, async_client: AsyncClient):
  527. """Verify 404 for non-existent printer."""
  528. response = await async_client.get("/api/v1/printers/99999/current-print-user")
  529. assert response.status_code == 404
  530. @pytest.mark.asyncio
  531. @pytest.mark.integration
  532. async def test_get_current_print_user_returns_empty_when_no_user(self, async_client: AsyncClient, printer_factory):
  533. """Verify empty object returned when no user is tracked."""
  534. printer = await printer_factory(name="Test Printer")
  535. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  536. mock_pm.get_current_print_user.return_value = None
  537. response = await async_client.get(f"/api/v1/printers/{printer.id}/current-print-user")
  538. assert response.status_code == 200
  539. assert response.json() == {}
  540. @pytest.mark.asyncio
  541. @pytest.mark.integration
  542. async def test_get_current_print_user_returns_user_info(self, async_client: AsyncClient, printer_factory):
  543. """Verify user info is returned when tracked."""
  544. printer = await printer_factory(name="Test Printer")
  545. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  546. mock_pm.get_current_print_user.return_value = {"user_id": 42, "username": "testuser"}
  547. response = await async_client.get(f"/api/v1/printers/{printer.id}/current-print-user")
  548. assert response.status_code == 200
  549. result = response.json()
  550. assert result["user_id"] == 42
  551. assert result["username"] == "testuser"
  552. class TestPrintControlAPI:
  553. """Integration tests for print control endpoints (stop, pause, resume)."""
  554. # ========================================================================
  555. # Stop print endpoint
  556. # ========================================================================
  557. @pytest.mark.asyncio
  558. @pytest.mark.integration
  559. async def test_stop_print_not_found(self, async_client: AsyncClient):
  560. """Verify 404 for non-existent printer."""
  561. response = await async_client.post("/api/v1/printers/99999/print/stop")
  562. assert response.status_code == 404
  563. @pytest.mark.asyncio
  564. @pytest.mark.integration
  565. async def test_stop_print_not_connected(self, async_client: AsyncClient, printer_factory):
  566. """Verify error when printer is not connected."""
  567. printer = await printer_factory(name="Disconnected Printer")
  568. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  569. mock_pm.get_client.return_value = None
  570. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/stop")
  571. assert response.status_code == 400
  572. assert "not connected" in response.json()["detail"].lower()
  573. @pytest.mark.asyncio
  574. @pytest.mark.integration
  575. async def test_stop_print_success(self, async_client: AsyncClient, printer_factory):
  576. """Verify successful stop print request."""
  577. printer = await printer_factory(name="Printing Printer")
  578. mock_client = MagicMock()
  579. mock_client.stop_print.return_value = True
  580. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  581. mock_pm.get_client.return_value = mock_client
  582. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/stop")
  583. assert response.status_code == 200
  584. assert response.json()["success"] is True
  585. mock_client.stop_print.assert_called_once()
  586. # ========================================================================
  587. # Pause print endpoint
  588. # ========================================================================
  589. @pytest.mark.asyncio
  590. @pytest.mark.integration
  591. async def test_pause_print_not_found(self, async_client: AsyncClient):
  592. """Verify 404 for non-existent printer."""
  593. response = await async_client.post("/api/v1/printers/99999/print/pause")
  594. assert response.status_code == 404
  595. @pytest.mark.asyncio
  596. @pytest.mark.integration
  597. async def test_pause_print_not_connected(self, async_client: AsyncClient, printer_factory):
  598. """Verify error when printer is not connected."""
  599. printer = await printer_factory(name="Disconnected Printer")
  600. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  601. mock_pm.get_client.return_value = None
  602. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/pause")
  603. assert response.status_code == 400
  604. assert "not connected" in response.json()["detail"].lower()
  605. @pytest.mark.asyncio
  606. @pytest.mark.integration
  607. async def test_pause_print_success(self, async_client: AsyncClient, printer_factory):
  608. """Verify successful pause print request."""
  609. printer = await printer_factory(name="Printing Printer")
  610. mock_client = MagicMock()
  611. mock_client.pause_print.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}/print/pause")
  615. assert response.status_code == 200
  616. assert response.json()["success"] is True
  617. mock_client.pause_print.assert_called_once()
  618. # ========================================================================
  619. # Resume print endpoint
  620. # ========================================================================
  621. @pytest.mark.asyncio
  622. @pytest.mark.integration
  623. async def test_resume_print_not_found(self, async_client: AsyncClient):
  624. """Verify 404 for non-existent printer."""
  625. response = await async_client.post("/api/v1/printers/99999/print/resume")
  626. assert response.status_code == 404
  627. @pytest.mark.asyncio
  628. @pytest.mark.integration
  629. async def test_resume_print_not_connected(self, async_client: AsyncClient, printer_factory):
  630. """Verify error when printer is not connected."""
  631. printer = await printer_factory(name="Disconnected Printer")
  632. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  633. mock_pm.get_client.return_value = None
  634. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/resume")
  635. assert response.status_code == 400
  636. assert "not connected" in response.json()["detail"].lower()
  637. @pytest.mark.asyncio
  638. @pytest.mark.integration
  639. async def test_resume_print_success(self, async_client: AsyncClient, printer_factory):
  640. """Verify successful resume print request."""
  641. printer = await printer_factory(name="Paused Printer")
  642. mock_client = MagicMock()
  643. mock_client.resume_print.return_value = True
  644. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  645. mock_pm.get_client.return_value = mock_client
  646. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/resume")
  647. assert response.status_code == 200
  648. assert response.json()["success"] is True
  649. mock_client.resume_print.assert_called_once()
  650. class TestAMSRefreshAPI:
  651. """Integration tests for AMS slot refresh endpoint."""
  652. @pytest.mark.asyncio
  653. @pytest.mark.integration
  654. async def test_ams_refresh_not_found(self, async_client: AsyncClient):
  655. """Verify 404 for non-existent printer."""
  656. response = await async_client.post("/api/v1/printers/99999/ams/0/slot/0/refresh")
  657. assert response.status_code == 404
  658. @pytest.mark.asyncio
  659. @pytest.mark.integration
  660. async def test_ams_refresh_not_connected(self, async_client: AsyncClient, printer_factory):
  661. """Verify error when printer is not connected."""
  662. printer = await printer_factory(name="Disconnected Printer")
  663. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  664. mock_pm.get_client.return_value = None
  665. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/0/slot/0/refresh")
  666. assert response.status_code == 400
  667. assert "not connected" in response.json()["detail"].lower()
  668. @pytest.mark.asyncio
  669. @pytest.mark.integration
  670. async def test_ams_refresh_success(self, async_client: AsyncClient, printer_factory):
  671. """Verify successful AMS refresh request."""
  672. printer = await printer_factory(name="Printer with AMS")
  673. mock_client = MagicMock()
  674. mock_client.ams_refresh_tray.return_value = (True, "Refreshing AMS 0 tray 1")
  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/0/slot/1/refresh")
  678. assert response.status_code == 200
  679. result = response.json()
  680. assert result["success"] is True
  681. mock_client.ams_refresh_tray.assert_called_once_with(0, 1)
  682. @pytest.mark.asyncio
  683. @pytest.mark.integration
  684. async def test_ams_refresh_filament_loaded(self, async_client: AsyncClient, printer_factory):
  685. """Verify error when filament is loaded (can't refresh while loaded)."""
  686. printer = await printer_factory(name="Printer with AMS")
  687. mock_client = MagicMock()
  688. mock_client.ams_refresh_tray.return_value = (False, "Please unload filament first")
  689. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  690. mock_pm.get_client.return_value = mock_client
  691. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/0/slot/0/refresh")
  692. assert response.status_code == 400
  693. assert "unload" in response.json()["detail"].lower()
  694. class TestAMSLoadUnloadAPI:
  695. """Integration tests for AMS load / unload endpoints (#891)."""
  696. # ── load ─────────────────────────────────────────────────────────────────
  697. @pytest.mark.asyncio
  698. @pytest.mark.integration
  699. async def test_load_invalid_tray_id(self, async_client: AsyncClient, printer_factory):
  700. """tray_id outside {0..15, 254, 255} is rejected."""
  701. printer = await printer_factory(name="P")
  702. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=99")
  703. assert response.status_code == 400
  704. @pytest.mark.asyncio
  705. @pytest.mark.integration
  706. async def test_load_not_found(self, async_client: AsyncClient):
  707. response = await async_client.post("/api/v1/printers/99999/ams/load?tray_id=0")
  708. assert response.status_code == 404
  709. @pytest.mark.asyncio
  710. @pytest.mark.integration
  711. async def test_load_not_connected(self, async_client: AsyncClient, printer_factory):
  712. printer = await printer_factory(name="Disconnected")
  713. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  714. mock_pm.get_client.return_value = None
  715. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=0")
  716. assert response.status_code == 400
  717. assert "not connected" in response.json()["detail"].lower()
  718. @pytest.mark.asyncio
  719. @pytest.mark.integration
  720. async def test_load_ams_slot_success(self, async_client: AsyncClient, printer_factory):
  721. """tray_id=5 → AMS 1 slot 2 (1-indexed in the message)."""
  722. printer = await printer_factory(name="P")
  723. mock_client = MagicMock()
  724. mock_client.ams_load_filament.return_value = True
  725. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  726. mock_pm.get_client.return_value = mock_client
  727. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=5")
  728. assert response.status_code == 200
  729. mock_client.ams_load_filament.assert_called_once_with(5)
  730. assert "AMS 1" in response.json()["message"]
  731. @pytest.mark.asyncio
  732. @pytest.mark.integration
  733. async def test_load_external_left_success(self, async_client: AsyncClient, printer_factory):
  734. """tray_id=254 → external spool / Ext-L."""
  735. printer = await printer_factory(name="P")
  736. mock_client = MagicMock()
  737. mock_client.ams_load_filament.return_value = True
  738. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  739. mock_pm.get_client.return_value = mock_client
  740. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=254")
  741. assert response.status_code == 200
  742. mock_client.ams_load_filament.assert_called_once_with(254)
  743. assert "external" in response.json()["message"].lower()
  744. @pytest.mark.asyncio
  745. @pytest.mark.integration
  746. async def test_load_external_right_success(self, async_client: AsyncClient, printer_factory):
  747. """tray_id=255 → Ext-R on dual-nozzle H2D."""
  748. printer = await printer_factory(name="H2D")
  749. mock_client = MagicMock()
  750. mock_client.ams_load_filament.return_value = True
  751. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  752. mock_pm.get_client.return_value = mock_client
  753. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=255")
  754. assert response.status_code == 200
  755. mock_client.ams_load_filament.assert_called_once_with(255)
  756. assert "Ext-R" in response.json()["message"]
  757. @pytest.mark.asyncio
  758. @pytest.mark.integration
  759. async def test_load_mqtt_failure_returns_500(self, async_client: AsyncClient, printer_factory):
  760. printer = await printer_factory(name="P")
  761. mock_client = MagicMock()
  762. mock_client.ams_load_filament.return_value = False
  763. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  764. mock_pm.get_client.return_value = mock_client
  765. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/load?tray_id=0")
  766. assert response.status_code == 500
  767. assert "failed" in response.json()["detail"].lower()
  768. # ── unload ───────────────────────────────────────────────────────────────
  769. @pytest.mark.asyncio
  770. @pytest.mark.integration
  771. async def test_unload_not_found(self, async_client: AsyncClient):
  772. response = await async_client.post("/api/v1/printers/99999/ams/unload")
  773. assert response.status_code == 404
  774. @pytest.mark.asyncio
  775. @pytest.mark.integration
  776. async def test_unload_not_connected(self, async_client: AsyncClient, printer_factory):
  777. printer = await printer_factory(name="Disconnected")
  778. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  779. mock_pm.get_client.return_value = None
  780. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/unload")
  781. assert response.status_code == 400
  782. assert "not connected" in response.json()["detail"].lower()
  783. @pytest.mark.asyncio
  784. @pytest.mark.integration
  785. async def test_unload_success(self, async_client: AsyncClient, printer_factory):
  786. printer = await printer_factory(name="P")
  787. mock_client = MagicMock()
  788. mock_client.ams_unload_filament.return_value = True
  789. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  790. mock_pm.get_client.return_value = mock_client
  791. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/unload")
  792. assert response.status_code == 200
  793. mock_client.ams_unload_filament.assert_called_once_with()
  794. assert response.json()["success"] is True
  795. @pytest.mark.asyncio
  796. @pytest.mark.integration
  797. async def test_unload_mqtt_failure_returns_500(self, async_client: AsyncClient, printer_factory):
  798. printer = await printer_factory(name="P")
  799. mock_client = MagicMock()
  800. mock_client.ams_unload_filament.return_value = False
  801. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  802. mock_pm.get_client.return_value = mock_client
  803. response = await async_client.post(f"/api/v1/printers/{printer.id}/ams/unload")
  804. assert response.status_code == 500
  805. assert "failed" in response.json()["detail"].lower()
  806. class TestConfigureAMSSlotAPI:
  807. """Integration tests for AMS slot configure endpoint — tray_info_idx resolution."""
  808. @pytest.mark.asyncio
  809. @pytest.mark.integration
  810. async def test_configure_not_connected(self, async_client: AsyncClient, printer_factory):
  811. """Verify error when printer is not connected."""
  812. printer = await printer_factory(name="Disconnected")
  813. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  814. mock_pm.get_client.return_value = None
  815. response = await async_client.post(
  816. f"/api/v1/printers/{printer.id}/slots/0/0/configure",
  817. params={
  818. "tray_info_idx": "GFL99",
  819. "tray_type": "PLA",
  820. "tray_sub_brands": "PLA Basic",
  821. "tray_color": "FF0000FF",
  822. "nozzle_temp_min": 190,
  823. "nozzle_temp_max": 230,
  824. },
  825. )
  826. assert response.status_code == 400
  827. assert "not connected" in response.json()["detail"].lower()
  828. @pytest.mark.asyncio
  829. @pytest.mark.integration
  830. async def test_configure_with_gf_id_keeps_it(self, async_client: AsyncClient, printer_factory):
  831. """Standard Bambu GF* filament IDs are sent as-is."""
  832. printer = await printer_factory(name="H2D")
  833. mock_client = MagicMock()
  834. mock_client.ams_set_filament_setting.return_value = True
  835. mock_client.extrusion_cali_sel.return_value = True
  836. mock_client.request_status_update.return_value = True
  837. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  838. mock_pm.get_client.return_value = mock_client
  839. mock_pm.get_status.return_value = None # No existing state
  840. response = await async_client.post(
  841. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  842. params={
  843. "tray_info_idx": "GFL05",
  844. "tray_type": "PLA",
  845. "tray_sub_brands": "PLA Basic",
  846. "tray_color": "FFFFFFFF",
  847. "nozzle_temp_min": 190,
  848. "nozzle_temp_max": 230,
  849. },
  850. )
  851. assert response.status_code == 200
  852. call_kwargs = mock_client.ams_set_filament_setting.call_args
  853. assert call_kwargs.kwargs["tray_info_idx"] == "GFL05"
  854. @pytest.mark.asyncio
  855. @pytest.mark.integration
  856. async def test_configure_pfus_sent_directly(self, async_client: AsyncClient, printer_factory):
  857. """PFUS* cloud-synced custom preset IDs are sent to the printer."""
  858. printer = await printer_factory(name="H2D")
  859. mock_client = MagicMock()
  860. mock_client.ams_set_filament_setting.return_value = True
  861. mock_client.extrusion_cali_sel.return_value = True
  862. mock_client.request_status_update.return_value = True
  863. mock_status = MagicMock()
  864. mock_status.raw_data = {"ams": {"ams": []}} # No existing tray data
  865. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  866. mock_pm.get_client.return_value = mock_client
  867. mock_pm.get_status.return_value = mock_status
  868. response = await async_client.post(
  869. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  870. params={
  871. "tray_info_idx": "PFUS9ac902733670a9",
  872. "tray_type": "PLA",
  873. "tray_sub_brands": "Devil Design PLA",
  874. "tray_color": "FF0000FF",
  875. "nozzle_temp_min": 190,
  876. "nozzle_temp_max": 230,
  877. },
  878. )
  879. assert response.status_code == 200
  880. call_kwargs = mock_client.ams_set_filament_setting.call_args
  881. assert call_kwargs.kwargs["tray_info_idx"] == "PFUS9ac902733670a9"
  882. @pytest.mark.asyncio
  883. @pytest.mark.integration
  884. async def test_configure_pfus_takes_priority_over_slot(self, async_client: AsyncClient, printer_factory):
  885. """Provided PFUS* preset takes priority over slot's existing preset."""
  886. printer = await printer_factory(name="H2D")
  887. mock_client = MagicMock()
  888. mock_client.ams_set_filament_setting.return_value = True
  889. mock_client.extrusion_cali_sel.return_value = True
  890. mock_client.request_status_update.return_value = True
  891. # Simulate slot already configured by slicer with cloud-synced preset
  892. mock_status = MagicMock()
  893. mock_status.raw_data = {
  894. "ams": {
  895. "ams": [
  896. {
  897. "id": 2,
  898. "tray": [
  899. {
  900. "id": 3,
  901. "tray_info_idx": "P4d64437",
  902. "tray_type": "PLA",
  903. "tray_color": "FF0000FF",
  904. }
  905. ],
  906. }
  907. ]
  908. }
  909. }
  910. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  911. mock_pm.get_client.return_value = mock_client
  912. mock_pm.get_status.return_value = mock_status
  913. response = await async_client.post(
  914. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  915. params={
  916. "tray_info_idx": "PFUS9ac902733670a9",
  917. "tray_type": "PLA",
  918. "tray_sub_brands": "Devil Design PLA",
  919. "tray_color": "FF0000FF",
  920. "nozzle_temp_min": 190,
  921. "nozzle_temp_max": 230,
  922. },
  923. )
  924. assert response.status_code == 200
  925. call_kwargs = mock_client.ams_set_filament_setting.call_args
  926. # Provided preset wins over slot's existing one
  927. assert call_kwargs.kwargs["tray_info_idx"] == "PFUS9ac902733670a9"
  928. @pytest.mark.asyncio
  929. @pytest.mark.integration
  930. async def test_configure_pfus_used_regardless_of_slot_material(self, async_client: AsyncClient, printer_factory):
  931. """Provided PFUS* preset is used even when slot has a different material."""
  932. printer = await printer_factory(name="H2D")
  933. mock_client = MagicMock()
  934. mock_client.ams_set_filament_setting.return_value = True
  935. mock_client.extrusion_cali_sel.return_value = True
  936. mock_client.request_status_update.return_value = True
  937. # Slot currently has PETG but user is configuring PLA
  938. mock_status = MagicMock()
  939. mock_status.raw_data = {
  940. "ams": {
  941. "ams": [
  942. {
  943. "id": 2,
  944. "tray": [{"id": 3, "tray_info_idx": "GFG99", "tray_type": "PETG", "tray_color": "FFFFFFFF"}],
  945. }
  946. ]
  947. }
  948. }
  949. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  950. mock_pm.get_client.return_value = mock_client
  951. mock_pm.get_status.return_value = mock_status
  952. response = await async_client.post(
  953. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  954. params={
  955. "tray_info_idx": "PFUS9ac902733670a9",
  956. "tray_type": "PLA",
  957. "tray_sub_brands": "Devil Design PLA",
  958. "tray_color": "FF0000FF",
  959. "nozzle_temp_min": 190,
  960. "nozzle_temp_max": 230,
  961. },
  962. )
  963. assert response.status_code == 200
  964. call_kwargs = mock_client.ams_set_filament_setting.call_args
  965. # Provided preset wins — slot's material is irrelevant
  966. assert call_kwargs.kwargs["tray_info_idx"] == "PFUS9ac902733670a9"
  967. @pytest.mark.asyncio
  968. @pytest.mark.integration
  969. async def test_configure_empty_id_uses_generic(self, async_client: AsyncClient, printer_factory):
  970. """Empty tray_info_idx (local preset) is replaced with generic."""
  971. printer = await printer_factory(name="H2D")
  972. mock_client = MagicMock()
  973. mock_client.ams_set_filament_setting.return_value = True
  974. mock_client.extrusion_cali_sel.return_value = True
  975. mock_client.request_status_update.return_value = True
  976. mock_status = MagicMock()
  977. mock_status.raw_data = {"ams": {"ams": []}}
  978. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  979. mock_pm.get_client.return_value = mock_client
  980. mock_pm.get_status.return_value = mock_status
  981. response = await async_client.post(
  982. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  983. params={
  984. "tray_info_idx": "",
  985. "tray_type": "PETG",
  986. "tray_sub_brands": "PETG Basic",
  987. "tray_color": "FFFFFFFF",
  988. "nozzle_temp_min": 220,
  989. "nozzle_temp_max": 260,
  990. },
  991. )
  992. assert response.status_code == 200
  993. call_kwargs = mock_client.ams_set_filament_setting.call_args
  994. assert call_kwargs.kwargs["tray_info_idx"] == "GFG99"
  995. @pytest.mark.asyncio
  996. @pytest.mark.integration
  997. async def test_configure_pfus_preserves_setting_id_pair(self, async_client: AsyncClient, printer_factory):
  998. """Both tray_info_idx=PFUS* and setting_id=PFUS* are forwarded untouched.
  999. Pins the end-to-end contract the frontend #1053 fix relies on: when the
  1000. user configures a slot with a custom cloud preset whose cloud detail
  1001. has filament_id=null, the frontend sends the setting_id in BOTH fields
  1002. and the backend must not collapse either to a generic GF* ID.
  1003. """
  1004. printer = await printer_factory(name="H2D")
  1005. mock_client = MagicMock()
  1006. mock_client.ams_set_filament_setting.return_value = True
  1007. mock_client.extrusion_cali_sel.return_value = True
  1008. mock_client.request_status_update.return_value = True
  1009. mock_status = MagicMock()
  1010. mock_status.raw_data = {"ams": {"ams": []}}
  1011. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1012. mock_pm.get_client.return_value = mock_client
  1013. mock_pm.get_status.return_value = mock_status
  1014. response = await async_client.post(
  1015. f"/api/v1/printers/{printer.id}/slots/128/0/configure",
  1016. params={
  1017. "tray_info_idx": "PFUSa8fb76f9733e3c",
  1018. "tray_type": "ABS",
  1019. "tray_sub_brands": "Sting3D ABS",
  1020. "tray_color": "000000FF",
  1021. "nozzle_temp_min": 240,
  1022. "nozzle_temp_max": 280,
  1023. "setting_id": "PFUSa8fb76f9733e3c",
  1024. },
  1025. )
  1026. assert response.status_code == 200
  1027. call_kwargs = mock_client.ams_set_filament_setting.call_args
  1028. assert call_kwargs.kwargs["tray_info_idx"] == "PFUSa8fb76f9733e3c"
  1029. assert call_kwargs.kwargs["setting_id"] == "PFUSa8fb76f9733e3c"
  1030. # Explicitly assert no generic-collapse happened for this HT slot.
  1031. assert call_kwargs.kwargs["tray_info_idx"] != "GFB99"
  1032. class TestSkipObjectsAPI:
  1033. """Integration tests for skip objects endpoints."""
  1034. # ========================================================================
  1035. # Get printable objects endpoint
  1036. # ========================================================================
  1037. @pytest.mark.asyncio
  1038. @pytest.mark.integration
  1039. async def test_get_objects_not_found(self, async_client: AsyncClient):
  1040. """Verify 404 for non-existent printer."""
  1041. response = await async_client.get("/api/v1/printers/99999/print/objects")
  1042. assert response.status_code == 404
  1043. @pytest.mark.asyncio
  1044. @pytest.mark.integration
  1045. async def test_get_objects_not_connected(self, async_client: AsyncClient, printer_factory):
  1046. """Verify error when printer is not connected."""
  1047. printer = await printer_factory(name="Disconnected Printer")
  1048. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1049. mock_pm.get_client.return_value = None
  1050. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  1051. assert response.status_code == 400
  1052. assert "not connected" in response.json()["detail"].lower()
  1053. @pytest.mark.asyncio
  1054. @pytest.mark.integration
  1055. async def test_get_objects_empty(self, async_client: AsyncClient, printer_factory):
  1056. """Verify empty objects list when no print is active."""
  1057. printer = await printer_factory(name="Idle Printer")
  1058. mock_client = MagicMock()
  1059. mock_client.state.printable_objects = {}
  1060. mock_client.state.skipped_objects = []
  1061. mock_client.state.state = "IDLE"
  1062. mock_client.state.subtask_name = None # Prevent FTP download attempt
  1063. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1064. mock_pm.get_client.return_value = mock_client
  1065. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  1066. assert response.status_code == 200
  1067. result = response.json()
  1068. assert result["objects"] == []
  1069. assert result["total"] == 0
  1070. assert result["skipped_count"] == 0
  1071. assert result["is_printing"] is False
  1072. @pytest.mark.asyncio
  1073. @pytest.mark.integration
  1074. async def test_get_objects_with_data(self, async_client: AsyncClient, printer_factory):
  1075. """Verify objects list when print is active."""
  1076. printer = await printer_factory(name="Printing Printer")
  1077. mock_client = MagicMock()
  1078. mock_client.state.printable_objects = {100: "Part A", 200: "Part B", 300: "Part C"}
  1079. mock_client.state.skipped_objects = [200]
  1080. mock_client.state.state = "RUNNING"
  1081. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1082. mock_pm.get_client.return_value = mock_client
  1083. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  1084. assert response.status_code == 200
  1085. result = response.json()
  1086. assert result["total"] == 3
  1087. assert result["skipped_count"] == 1
  1088. assert result["is_printing"] is True
  1089. # Check objects have correct structure
  1090. objects_by_id = {obj["id"]: obj for obj in result["objects"]}
  1091. assert objects_by_id[100]["name"] == "Part A"
  1092. assert objects_by_id[100]["skipped"] is False
  1093. assert objects_by_id[200]["name"] == "Part B"
  1094. assert objects_by_id[200]["skipped"] is True
  1095. assert objects_by_id[300]["name"] == "Part C"
  1096. assert objects_by_id[300]["skipped"] is False
  1097. # ========================================================================
  1098. # Skip objects endpoint
  1099. # ========================================================================
  1100. @pytest.mark.asyncio
  1101. @pytest.mark.integration
  1102. async def test_get_objects_with_positions(self, async_client: AsyncClient, printer_factory):
  1103. """Verify objects list includes position data when available."""
  1104. printer = await printer_factory(name="Printing Printer")
  1105. # New format with position data
  1106. mock_client = MagicMock()
  1107. mock_client.state.printable_objects = {
  1108. 100: {"name": "Part A", "x": 50.0, "y": 100.0},
  1109. 200: {"name": "Part B", "x": 150.0, "y": 100.0},
  1110. }
  1111. mock_client.state.skipped_objects = []
  1112. mock_client.state.state = "RUNNING"
  1113. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1114. mock_pm.get_client.return_value = mock_client
  1115. response = await async_client.get(f"/api/v1/printers/{printer.id}/print/objects")
  1116. assert response.status_code == 200
  1117. result = response.json()
  1118. assert result["total"] == 2
  1119. # Check objects have position data
  1120. objects_by_id = {obj["id"]: obj for obj in result["objects"]}
  1121. assert objects_by_id[100]["name"] == "Part A"
  1122. assert objects_by_id[100]["x"] == 50.0
  1123. assert objects_by_id[100]["y"] == 100.0
  1124. assert objects_by_id[200]["name"] == "Part B"
  1125. assert objects_by_id[200]["x"] == 150.0
  1126. assert objects_by_id[200]["y"] == 100.0
  1127. # ========================================================================
  1128. # Skip objects endpoint
  1129. # ========================================================================
  1130. @pytest.mark.asyncio
  1131. @pytest.mark.integration
  1132. async def test_skip_objects_not_found(self, async_client: AsyncClient):
  1133. """Verify 404 for non-existent printer."""
  1134. response = await async_client.post("/api/v1/printers/99999/print/skip-objects", json=[100])
  1135. assert response.status_code == 404
  1136. @pytest.mark.asyncio
  1137. @pytest.mark.integration
  1138. async def test_skip_objects_not_connected(self, async_client: AsyncClient, printer_factory):
  1139. """Verify error when printer is not connected."""
  1140. printer = await printer_factory(name="Disconnected Printer")
  1141. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1142. mock_pm.get_client.return_value = None
  1143. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[100])
  1144. assert response.status_code == 400
  1145. assert "not connected" in response.json()["detail"].lower()
  1146. @pytest.mark.asyncio
  1147. @pytest.mark.integration
  1148. async def test_skip_objects_empty_list(self, async_client: AsyncClient, printer_factory):
  1149. """Verify error when no object IDs provided."""
  1150. printer = await printer_factory(name="Printing Printer")
  1151. mock_client = MagicMock()
  1152. mock_client.state.printable_objects = {100: "Part A"}
  1153. mock_client.state.skipped_objects = []
  1154. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1155. mock_pm.get_client.return_value = mock_client
  1156. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[])
  1157. assert response.status_code == 400
  1158. assert "no object" in response.json()["detail"].lower()
  1159. @pytest.mark.asyncio
  1160. @pytest.mark.integration
  1161. async def test_skip_objects_invalid_id(self, async_client: AsyncClient, printer_factory):
  1162. """Verify error when object ID doesn't exist."""
  1163. printer = await printer_factory(name="Printing Printer")
  1164. mock_client = MagicMock()
  1165. mock_client.state.printable_objects = {100: "Part A"}
  1166. mock_client.state.skipped_objects = []
  1167. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1168. mock_pm.get_client.return_value = mock_client
  1169. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[999])
  1170. assert response.status_code == 400
  1171. assert "invalid" in response.json()["detail"].lower()
  1172. @pytest.mark.asyncio
  1173. @pytest.mark.integration
  1174. async def test_skip_objects_success(self, async_client: AsyncClient, printer_factory):
  1175. """Verify successful skip objects request."""
  1176. printer = await printer_factory(name="Printing Printer")
  1177. mock_client = MagicMock()
  1178. mock_client.state.printable_objects = {100: "Part A", 200: "Part B"}
  1179. mock_client.state.skipped_objects = []
  1180. mock_client.skip_objects.return_value = True
  1181. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1182. mock_pm.get_client.return_value = mock_client
  1183. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[100])
  1184. assert response.status_code == 200
  1185. result = response.json()
  1186. assert result["success"] is True
  1187. assert 100 in result["skipped_objects"]
  1188. mock_client.skip_objects.assert_called_once_with([100])
  1189. @pytest.mark.asyncio
  1190. @pytest.mark.integration
  1191. async def test_skip_objects_multiple(self, async_client: AsyncClient, printer_factory):
  1192. """Verify skipping multiple objects at once."""
  1193. printer = await printer_factory(name="Printing Printer")
  1194. mock_client = MagicMock()
  1195. mock_client.state.printable_objects = {100: "Part A", 200: "Part B", 300: "Part C"}
  1196. mock_client.state.skipped_objects = []
  1197. mock_client.skip_objects.return_value = True
  1198. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1199. mock_pm.get_client.return_value = mock_client
  1200. response = await async_client.post(f"/api/v1/printers/{printer.id}/print/skip-objects", json=[100, 200])
  1201. assert response.status_code == 200
  1202. result = response.json()
  1203. assert result["success"] is True
  1204. assert 100 in result["skipped_objects"]
  1205. assert 200 in result["skipped_objects"]
  1206. mock_client.skip_objects.assert_called_once_with([100, 200])
  1207. class TestChamberLightAPI:
  1208. """Integration tests for chamber light control endpoint."""
  1209. @pytest.mark.asyncio
  1210. @pytest.mark.integration
  1211. async def test_chamber_light_not_found(self, async_client: AsyncClient):
  1212. """Verify 404 for non-existent printer."""
  1213. response = await async_client.post("/api/v1/printers/99999/chamber-light?on=true")
  1214. assert response.status_code == 404
  1215. @pytest.mark.asyncio
  1216. @pytest.mark.integration
  1217. async def test_chamber_light_not_connected(self, async_client: AsyncClient, printer_factory):
  1218. """Verify error when printer is not connected."""
  1219. printer = await printer_factory(name="Disconnected Printer")
  1220. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1221. mock_pm.get_client.return_value = None
  1222. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=true")
  1223. assert response.status_code == 400
  1224. assert "not connected" in response.json()["detail"].lower()
  1225. @pytest.mark.asyncio
  1226. @pytest.mark.integration
  1227. async def test_chamber_light_on_success(self, async_client: AsyncClient, printer_factory):
  1228. """Verify successful chamber light on request."""
  1229. printer = await printer_factory(name="Test Printer")
  1230. mock_client = MagicMock()
  1231. mock_client.set_chamber_light.return_value = True
  1232. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1233. mock_pm.get_client.return_value = mock_client
  1234. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=true")
  1235. assert response.status_code == 200
  1236. result = response.json()
  1237. assert result["success"] is True
  1238. assert "on" in result["message"].lower()
  1239. mock_client.set_chamber_light.assert_called_once_with(True)
  1240. @pytest.mark.asyncio
  1241. @pytest.mark.integration
  1242. async def test_chamber_light_off_success(self, async_client: AsyncClient, printer_factory):
  1243. """Verify successful chamber light off request."""
  1244. printer = await printer_factory(name="Test Printer")
  1245. mock_client = MagicMock()
  1246. mock_client.set_chamber_light.return_value = True
  1247. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1248. mock_pm.get_client.return_value = mock_client
  1249. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=false")
  1250. assert response.status_code == 200
  1251. result = response.json()
  1252. assert result["success"] is True
  1253. assert "off" in result["message"].lower()
  1254. mock_client.set_chamber_light.assert_called_once_with(False)
  1255. @pytest.mark.asyncio
  1256. @pytest.mark.integration
  1257. async def test_chamber_light_failure(self, async_client: AsyncClient, printer_factory):
  1258. """Verify error handling when chamber light control fails."""
  1259. printer = await printer_factory(name="Test Printer")
  1260. mock_client = MagicMock()
  1261. mock_client.set_chamber_light.return_value = False
  1262. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1263. mock_pm.get_client.return_value = mock_client
  1264. response = await async_client.post(f"/api/v1/printers/{printer.id}/chamber-light?on=true")
  1265. assert response.status_code == 500
  1266. assert "failed" in response.json()["detail"].lower()
  1267. class TestAirductModeAPI:
  1268. """Integration tests for the airduct mode endpoint (P2S/H2*)."""
  1269. @pytest.mark.asyncio
  1270. @pytest.mark.integration
  1271. async def test_invalid_mode_rejected(self, async_client: AsyncClient, printer_factory):
  1272. printer = await printer_factory(name="P", model="P2S")
  1273. response = await async_client.post(f"/api/v1/printers/{printer.id}/airduct-mode?mode=foo")
  1274. assert response.status_code == 400
  1275. @pytest.mark.asyncio
  1276. @pytest.mark.integration
  1277. async def test_not_connected(self, async_client: AsyncClient, printer_factory):
  1278. printer = await printer_factory(name="P", model="P2S")
  1279. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1280. mock_pm.get_client.return_value = None
  1281. response = await async_client.post(f"/api/v1/printers/{printer.id}/airduct-mode?mode=cooling")
  1282. assert response.status_code == 400
  1283. @pytest.mark.asyncio
  1284. @pytest.mark.integration
  1285. async def test_cooling_success(self, async_client: AsyncClient, printer_factory):
  1286. printer = await printer_factory(name="P", model="P2S")
  1287. mock_client = MagicMock()
  1288. mock_client.set_airduct_mode.return_value = True
  1289. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1290. mock_pm.get_client.return_value = mock_client
  1291. response = await async_client.post(f"/api/v1/printers/{printer.id}/airduct-mode?mode=cooling")
  1292. assert response.status_code == 200
  1293. assert response.json()["success"] is True
  1294. mock_client.set_airduct_mode.assert_called_once_with("cooling")
  1295. @pytest.mark.asyncio
  1296. @pytest.mark.integration
  1297. async def test_heating_failure_returns_500(self, async_client: AsyncClient, printer_factory):
  1298. printer = await printer_factory(name="P", model="P2S")
  1299. mock_client = MagicMock()
  1300. mock_client.set_airduct_mode.return_value = False
  1301. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1302. mock_pm.get_client.return_value = mock_client
  1303. response = await async_client.post(f"/api/v1/printers/{printer.id}/airduct-mode?mode=heating")
  1304. assert response.status_code == 500
  1305. class TestClearHMSErrorsAPI:
  1306. """Integration tests for clear HMS errors endpoint."""
  1307. @pytest.mark.asyncio
  1308. @pytest.mark.integration
  1309. async def test_clear_hms_errors_not_found(self, async_client: AsyncClient):
  1310. """Verify 404 for non-existent printer."""
  1311. response = await async_client.post("/api/v1/printers/99999/hms/clear")
  1312. assert response.status_code == 404
  1313. @pytest.mark.asyncio
  1314. @pytest.mark.integration
  1315. async def test_clear_hms_errors_not_connected(self, async_client: AsyncClient, printer_factory):
  1316. """Verify error when printer is not connected."""
  1317. printer = await printer_factory(name="Disconnected Printer")
  1318. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1319. mock_pm.get_client.return_value = None
  1320. response = await async_client.post(f"/api/v1/printers/{printer.id}/hms/clear")
  1321. assert response.status_code == 400
  1322. assert "not connected" in response.json()["detail"].lower()
  1323. @pytest.mark.asyncio
  1324. @pytest.mark.integration
  1325. async def test_clear_hms_errors_success(self, async_client: AsyncClient, printer_factory):
  1326. """Verify successful clear HMS errors request."""
  1327. printer = await printer_factory(name="Test Printer")
  1328. mock_client = MagicMock()
  1329. mock_client.clear_hms_errors.return_value = True
  1330. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1331. mock_pm.get_client.return_value = mock_client
  1332. response = await async_client.post(f"/api/v1/printers/{printer.id}/hms/clear")
  1333. assert response.status_code == 200
  1334. result = response.json()
  1335. assert result["success"] is True
  1336. assert "cleared" in result["message"].lower()
  1337. mock_client.clear_hms_errors.assert_called_once()
  1338. @pytest.mark.asyncio
  1339. @pytest.mark.integration
  1340. async def test_clear_hms_errors_failure(self, async_client: AsyncClient, printer_factory):
  1341. """Verify error handling when clear HMS errors fails."""
  1342. printer = await printer_factory(name="Test Printer")
  1343. mock_client = MagicMock()
  1344. mock_client.clear_hms_errors.return_value = False
  1345. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1346. mock_pm.get_client.return_value = mock_client
  1347. response = await async_client.post(f"/api/v1/printers/{printer.id}/hms/clear")
  1348. assert response.status_code == 500
  1349. assert "failed" in response.json()["detail"].lower()
  1350. def _build_h2d_state(*, ams_id: int = 0, tray_id: int = 2, cali_idx: int = 5):
  1351. """Build a MagicMock PrinterState for an H2D printer with a single BL spool tray.
  1352. Used by both TestApplyPaAfterRefresh (Phase 13 P13-T-BE-1) and the K-profile
  1353. persistence tests below. The tray data passes is_bambu_tag (32-char non-zero
  1354. tray_uuid + non-empty tray_info_idx).
  1355. """
  1356. nozzle = MagicMock(nozzle_diameter="0.4")
  1357. state = MagicMock()
  1358. state.nozzles = [nozzle]
  1359. state.ams_extruder_map = {"0": 0}
  1360. state.raw_data = {
  1361. "ams": [
  1362. {
  1363. "id": ams_id,
  1364. "tray": [
  1365. {
  1366. "id": tray_id,
  1367. "tray_type": "PLA",
  1368. "tag_uid": "AABBCC1122334400",
  1369. "tray_uuid": "11223344556677880011223344556677",
  1370. "tray_info_idx": "GFL05",
  1371. "cali_idx": cali_idx,
  1372. }
  1373. ],
  1374. }
  1375. ]
  1376. }
  1377. return state
  1378. def _patch_async_session_to(db_session):
  1379. """Patch backend.app.core.database.async_session so calls inside the function
  1380. under test reuse the test fixture's db_session.
  1381. `_apply_pa_after_refresh` lazy-imports `from backend.app.core.database import
  1382. async_session` at runtime (line 2849). When we patch the source module
  1383. before the call, the lazy import picks up the patched object.
  1384. Returns the patch context manager; use as `with _patch_async_session_to(db_session):`.
  1385. Pattern verified against test_print_lifecycle.py:38-42.
  1386. """
  1387. cm = AsyncMock()
  1388. cm.__aenter__ = AsyncMock(return_value=db_session)
  1389. cm.__aexit__ = AsyncMock(return_value=None)
  1390. return patch("backend.app.core.database.async_session", return_value=cm)
  1391. class TestApplyPaAfterRefresh:
  1392. """Phase 13 P13-T-BE-1: _apply_pa_after_refresh K-profile cascade.
  1393. Verifies the 3-stage cascade (local SpoolKProfile → Spoolman SpoolmanKProfile
  1394. → live tray.cali_idx fallback) and the Bug A regression (kp.extruder, not
  1395. kp.extruder_id, after the Phase 13 P13-2a fix).
  1396. `_apply_pa_after_refresh` is a free function spawned via asyncio.create_task
  1397. from the /ams-refresh endpoint. Tests call it directly because awaiting the
  1398. spawned task in an HTTP test would require sleeping past the 5-second guard
  1399. that delays MQTT until RFID re-read finishes.
  1400. """
  1401. @pytest.mark.asyncio
  1402. @pytest.mark.integration
  1403. async def test_local_kp_match_sends_stored_cali_idx(self, db_session, printer_factory):
  1404. """Local SpoolAssignment + matching SpoolKProfile → stored cali_idx wins."""
  1405. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1406. from backend.app.models.spool import Spool
  1407. from backend.app.models.spool_assignment import SpoolAssignment
  1408. from backend.app.models.spool_k_profile import SpoolKProfile
  1409. printer = await printer_factory()
  1410. spool = Spool(material="PLA", color_name="Red", rgba="FF0000FF")
  1411. db_session.add(spool)
  1412. await db_session.flush()
  1413. db_session.add(
  1414. SpoolAssignment(
  1415. spool_id=spool.id,
  1416. printer_id=printer.id,
  1417. ams_id=0,
  1418. tray_id=2,
  1419. )
  1420. )
  1421. db_session.add(
  1422. SpoolKProfile(
  1423. spool_id=spool.id,
  1424. printer_id=printer.id,
  1425. extruder=0,
  1426. nozzle_diameter="0.4",
  1427. k_value=0.025,
  1428. cali_idx=42,
  1429. )
  1430. )
  1431. await db_session.commit()
  1432. mock_client = MagicMock()
  1433. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1434. state = _build_h2d_state(cali_idx=5)
  1435. with (
  1436. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1437. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1438. _patch_async_session_to(db_session),
  1439. ):
  1440. mock_pm.get_client.return_value = mock_client
  1441. mock_pm.get_status.return_value = state
  1442. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1443. mock_client.extrusion_cali_sel.assert_called_once()
  1444. kwargs = mock_client.extrusion_cali_sel.call_args.kwargs
  1445. assert kwargs["cali_idx"] == 42 # stored profile, not 5 (live)
  1446. @pytest.mark.asyncio
  1447. @pytest.mark.integration
  1448. async def test_local_no_kp_uses_live_cali_idx(self, db_session, printer_factory):
  1449. """Local SpoolAssignment but no matching SpoolKProfile → live cali_idx (Stage 3)."""
  1450. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1451. from backend.app.models.spool import Spool
  1452. from backend.app.models.spool_assignment import SpoolAssignment
  1453. printer = await printer_factory()
  1454. spool = Spool(material="PLA")
  1455. db_session.add(spool)
  1456. await db_session.flush()
  1457. db_session.add(
  1458. SpoolAssignment(
  1459. spool_id=spool.id,
  1460. printer_id=printer.id,
  1461. ams_id=0,
  1462. tray_id=2,
  1463. )
  1464. )
  1465. await db_session.commit()
  1466. mock_client = MagicMock()
  1467. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1468. state = _build_h2d_state(cali_idx=5)
  1469. with (
  1470. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1471. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1472. _patch_async_session_to(db_session),
  1473. ):
  1474. mock_pm.get_client.return_value = mock_client
  1475. mock_pm.get_status.return_value = state
  1476. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1477. mock_client.extrusion_cali_sel.assert_called_once()
  1478. kwargs = mock_client.extrusion_cali_sel.call_args.kwargs
  1479. assert kwargs["cali_idx"] == 5 # live tray.cali_idx fallback
  1480. @pytest.mark.asyncio
  1481. @pytest.mark.integration
  1482. async def test_spoolman_kp_when_no_local(self, db_session, printer_factory):
  1483. """No local assignment + Spoolman SlotAssignment + SpoolmanKProfile → Spoolman cali_idx."""
  1484. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1485. from backend.app.models.spoolman_k_profile import SpoolmanKProfile
  1486. from backend.app.models.spoolman_slot_assignment import SpoolmanSlotAssignment
  1487. printer = await printer_factory()
  1488. db_session.add(
  1489. SpoolmanSlotAssignment(
  1490. printer_id=printer.id,
  1491. ams_id=0,
  1492. tray_id=2,
  1493. spoolman_spool_id=99,
  1494. )
  1495. )
  1496. db_session.add(
  1497. SpoolmanKProfile(
  1498. spoolman_spool_id=99,
  1499. printer_id=printer.id,
  1500. extruder=0,
  1501. nozzle_diameter="0.4",
  1502. k_value=0.030,
  1503. cali_idx=77,
  1504. )
  1505. )
  1506. await db_session.commit()
  1507. mock_client = MagicMock()
  1508. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1509. state = _build_h2d_state(cali_idx=5)
  1510. with (
  1511. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1512. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1513. _patch_async_session_to(db_session),
  1514. ):
  1515. mock_pm.get_client.return_value = mock_client
  1516. mock_pm.get_status.return_value = state
  1517. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1518. mock_client.extrusion_cali_sel.assert_called_once()
  1519. kwargs = mock_client.extrusion_cali_sel.call_args.kwargs
  1520. assert kwargs["cali_idx"] == 77 # Spoolman stored profile
  1521. @pytest.mark.asyncio
  1522. @pytest.mark.integration
  1523. async def test_spoolman_no_kp_uses_live(self, db_session, printer_factory):
  1524. """Spoolman SlotAssignment but no SpoolmanKProfile → live cali_idx (Stage 3)."""
  1525. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1526. from backend.app.models.spoolman_slot_assignment import SpoolmanSlotAssignment
  1527. printer = await printer_factory()
  1528. db_session.add(
  1529. SpoolmanSlotAssignment(
  1530. printer_id=printer.id,
  1531. ams_id=0,
  1532. tray_id=2,
  1533. spoolman_spool_id=99,
  1534. )
  1535. )
  1536. await db_session.commit()
  1537. mock_client = MagicMock()
  1538. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1539. state = _build_h2d_state(cali_idx=5)
  1540. with (
  1541. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1542. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1543. _patch_async_session_to(db_session),
  1544. ):
  1545. mock_pm.get_client.return_value = mock_client
  1546. mock_pm.get_status.return_value = state
  1547. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1548. mock_client.extrusion_cali_sel.assert_called_once()
  1549. assert mock_client.extrusion_cali_sel.call_args.kwargs["cali_idx"] == 5
  1550. @pytest.mark.asyncio
  1551. @pytest.mark.integration
  1552. async def test_no_assignment_uses_live(self, db_session, printer_factory):
  1553. """No assignment of any kind + live cali_idx >= 0 → live fallback."""
  1554. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1555. printer = await printer_factory()
  1556. mock_client = MagicMock()
  1557. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1558. state = _build_h2d_state(cali_idx=5)
  1559. with (
  1560. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1561. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1562. _patch_async_session_to(db_session),
  1563. ):
  1564. mock_pm.get_client.return_value = mock_client
  1565. mock_pm.get_status.return_value = state
  1566. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1567. mock_client.extrusion_cali_sel.assert_called_once()
  1568. assert mock_client.extrusion_cali_sel.call_args.kwargs["cali_idx"] == 5
  1569. @pytest.mark.asyncio
  1570. @pytest.mark.integration
  1571. async def test_negative_live_cali_idx_skipped(self, db_session, printer_factory):
  1572. """No assignment + live cali_idx=-1 → no MQTT call (invalid live value)."""
  1573. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1574. printer = await printer_factory()
  1575. mock_client = MagicMock()
  1576. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1577. state = _build_h2d_state(cali_idx=-1)
  1578. with (
  1579. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1580. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1581. _patch_async_session_to(db_session),
  1582. ):
  1583. mock_pm.get_client.return_value = mock_client
  1584. mock_pm.get_status.return_value = state
  1585. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1586. mock_client.extrusion_cali_sel.assert_not_called()
  1587. @pytest.mark.asyncio
  1588. @pytest.mark.integration
  1589. async def test_no_assignment_no_live_cali_idx_no_call(self, db_session, printer_factory):
  1590. """No assignment of any kind AND no live cali_idx in tray → no MQTT call.
  1591. Distinct from test_negative_live_cali_idx_skipped: that test has
  1592. cali_idx=-1 in raw_data; this one omits the field entirely (returns
  1593. None from .get("cali_idx")). Both must result in no MQTT call.
  1594. """
  1595. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1596. printer = await printer_factory()
  1597. mock_client = MagicMock()
  1598. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1599. # State with NO cali_idx field on tray at all
  1600. nozzle = MagicMock(nozzle_diameter="0.4")
  1601. state = MagicMock()
  1602. state.nozzles = [nozzle]
  1603. state.ams_extruder_map = {"0": 0}
  1604. state.raw_data = {
  1605. "ams": [
  1606. {
  1607. "id": 0,
  1608. "tray": [
  1609. {
  1610. "id": 2,
  1611. "tray_type": "PLA",
  1612. "tag_uid": "AABBCC1122334400",
  1613. "tray_uuid": "11223344556677880011223344556677",
  1614. "tray_info_idx": "GFL05",
  1615. # cali_idx field intentionally omitted
  1616. }
  1617. ],
  1618. }
  1619. ]
  1620. }
  1621. with (
  1622. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1623. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1624. _patch_async_session_to(db_session),
  1625. ):
  1626. mock_pm.get_client.return_value = mock_client
  1627. mock_pm.get_status.return_value = state
  1628. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1629. mock_client.extrusion_cali_sel.assert_not_called()
  1630. @pytest.mark.asyncio
  1631. @pytest.mark.integration
  1632. async def test_extruder_mismatch_uses_kp_as_fallback(self, db_session, printer_factory):
  1633. """K-profile for extruder=1 but slot is extruder=0 → no exact match,
  1634. but the kp is used as extruder-agnostic fallback rather than dropped.
  1635. Hard-skipping on extruder mismatch was the previous behavior; in
  1636. practice it caused stored K-profiles to be silently ignored whenever
  1637. the AMS-extruder mapping had shifted (or when only one of the two
  1638. extruders was ever calibrated for a given spool). The cascade now
  1639. prefers an exact extruder match but falls back to any matching kp
  1640. for the same printer + nozzle.
  1641. """
  1642. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1643. from backend.app.models.spool import Spool
  1644. from backend.app.models.spool_assignment import SpoolAssignment
  1645. from backend.app.models.spool_k_profile import SpoolKProfile
  1646. printer = await printer_factory()
  1647. spool = Spool(material="PLA")
  1648. db_session.add(spool)
  1649. await db_session.flush()
  1650. db_session.add(
  1651. SpoolAssignment(
  1652. spool_id=spool.id,
  1653. printer_id=printer.id,
  1654. ams_id=0,
  1655. tray_id=2,
  1656. )
  1657. )
  1658. # K-profile is for extruder=1, but slot's ams_extruder_map["0"]=0
  1659. db_session.add(
  1660. SpoolKProfile(
  1661. spool_id=spool.id,
  1662. printer_id=printer.id,
  1663. extruder=1,
  1664. nozzle_diameter="0.4",
  1665. k_value=0.025,
  1666. cali_idx=42,
  1667. )
  1668. )
  1669. await db_session.commit()
  1670. mock_client = MagicMock()
  1671. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1672. state = _build_h2d_state(cali_idx=5)
  1673. with (
  1674. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1675. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1676. _patch_async_session_to(db_session),
  1677. ):
  1678. mock_pm.get_client.return_value = mock_client
  1679. mock_pm.get_status.return_value = state
  1680. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1681. # No exact extruder match, but the stored kp wins as the
  1682. # extruder-agnostic fallback over live cali_idx=5.
  1683. mock_client.extrusion_cali_sel.assert_called_once()
  1684. assert mock_client.extrusion_cali_sel.call_args.kwargs["cali_idx"] == 42
  1685. @pytest.mark.asyncio
  1686. @pytest.mark.integration
  1687. async def test_extruder_exact_match_preferred_over_fallback(
  1688. self,
  1689. db_session,
  1690. printer_factory,
  1691. ):
  1692. """When two kp rows exist, one with matching extruder and one without,
  1693. the exact-extruder kp wins (extruder-agnostic fallback only fires when
  1694. no exact match exists).
  1695. """
  1696. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1697. from backend.app.models.spool import Spool
  1698. from backend.app.models.spool_assignment import SpoolAssignment
  1699. from backend.app.models.spool_k_profile import SpoolKProfile
  1700. printer = await printer_factory()
  1701. spool = Spool(material="PLA")
  1702. db_session.add(spool)
  1703. await db_session.flush()
  1704. db_session.add(
  1705. SpoolAssignment(
  1706. spool_id=spool.id,
  1707. printer_id=printer.id,
  1708. ams_id=0,
  1709. tray_id=2,
  1710. )
  1711. )
  1712. # Two kp rows: extruder=1 (mismatch w/ slot extruder=0) and extruder=0 (exact)
  1713. db_session.add(
  1714. SpoolKProfile(
  1715. spool_id=spool.id,
  1716. printer_id=printer.id,
  1717. extruder=1,
  1718. nozzle_diameter="0.4",
  1719. k_value=0.030,
  1720. cali_idx=99,
  1721. )
  1722. )
  1723. db_session.add(
  1724. SpoolKProfile(
  1725. spool_id=spool.id,
  1726. printer_id=printer.id,
  1727. extruder=0,
  1728. nozzle_diameter="0.4",
  1729. k_value=0.025,
  1730. cali_idx=42,
  1731. )
  1732. )
  1733. await db_session.commit()
  1734. mock_client = MagicMock()
  1735. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1736. state = _build_h2d_state(cali_idx=5)
  1737. with (
  1738. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1739. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1740. _patch_async_session_to(db_session),
  1741. ):
  1742. mock_pm.get_client.return_value = mock_client
  1743. mock_pm.get_status.return_value = state
  1744. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1745. # Exact-extruder=0 kp wins (cali_idx=42), not the extruder=1 fallback (99)
  1746. mock_client.extrusion_cali_sel.assert_called_once()
  1747. assert mock_client.extrusion_cali_sel.call_args.kwargs["cali_idx"] == 42
  1748. @pytest.mark.asyncio
  1749. @pytest.mark.integration
  1750. async def test_regression_bug_a_kp_extruder_attr(self, db_session, printer_factory):
  1751. """Regression test for Phase 13 P13-2a Bug A.
  1752. Pre-fix Z.2910 used `kp.extruder_id` (AttributeError on SpoolKProfile,
  1753. silently swallowed by outer try/except). On dual-nozzle printers with
  1754. slot_extruder != None this caused the K-profile match loop to crash.
  1755. After P13-2a the field name is correct: `kp.extruder`.
  1756. """
  1757. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1758. from backend.app.models.spool import Spool
  1759. from backend.app.models.spool_assignment import SpoolAssignment
  1760. from backend.app.models.spool_k_profile import SpoolKProfile
  1761. printer = await printer_factory()
  1762. spool = Spool(material="PLA")
  1763. db_session.add(spool)
  1764. await db_session.flush()
  1765. db_session.add(
  1766. SpoolAssignment(
  1767. spool_id=spool.id,
  1768. printer_id=printer.id,
  1769. ams_id=0,
  1770. tray_id=2,
  1771. )
  1772. )
  1773. # extruder=0 matches slot_extruder=0 (from ams_extruder_map={"0":0})
  1774. db_session.add(
  1775. SpoolKProfile(
  1776. spool_id=spool.id,
  1777. printer_id=printer.id,
  1778. extruder=0,
  1779. nozzle_diameter="0.4",
  1780. k_value=0.025,
  1781. cali_idx=42,
  1782. )
  1783. )
  1784. await db_session.commit()
  1785. mock_client = MagicMock()
  1786. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1787. state = _build_h2d_state(cali_idx=5)
  1788. with (
  1789. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1790. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1791. _patch_async_session_to(db_session),
  1792. ):
  1793. mock_pm.get_client.return_value = mock_client
  1794. mock_pm.get_status.return_value = state
  1795. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1796. # If Bug A regressed (kp.extruder_id), the loop would AttributeError → silent fail
  1797. # → no extrusion_cali_sel call. Post-fix the loop matches and sends cali_idx=42.
  1798. mock_client.extrusion_cali_sel.assert_called_once()
  1799. assert mock_client.extrusion_cali_sel.call_args.kwargs["cali_idx"] == 42
  1800. @pytest.mark.asyncio
  1801. @pytest.mark.integration
  1802. async def test_tag_fallback_finds_spool_when_assignment_missing(
  1803. self,
  1804. db_session,
  1805. printer_factory,
  1806. ):
  1807. """Stage 1b regression for the maintainer's #2 reproducer on H2D:
  1808. reset slot, trigger re-read → slot ends up on the default K-profile
  1809. instead of the spool's stored profile.
  1810. Setup mirrors the bug:
  1811. - Spool has tray_uuid set (the RFID tag was registered earlier).
  1812. - SpoolKProfile exists for that spool with cali_idx=42.
  1813. - NO SpoolAssignment row — the reset deleted it before the re-read
  1814. triggered _apply_pa_after_refresh, and tag-auto-detect has not
  1815. re-created it yet within the 5 s sleep window.
  1816. - Live tray.cali_idx=5 (firmware-default after the RFID re-read).
  1817. Without Stage 1b the cascade falls through to Stage 3 and re-asserts
  1818. the firmware-default cali_idx=5. With Stage 1b it locates the spool by
  1819. the live tray's tray_uuid and applies the stored cali_idx=42.
  1820. """
  1821. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1822. from backend.app.models.spool import Spool
  1823. from backend.app.models.spool_k_profile import SpoolKProfile
  1824. printer = await printer_factory()
  1825. # Spool with tray_uuid matching the one _build_h2d_state puts on the tray
  1826. spool = Spool(
  1827. material="PLA",
  1828. color_name="Red",
  1829. rgba="FF0000FF",
  1830. tray_uuid="11223344556677880011223344556677",
  1831. tag_uid="AABBCC1122334400",
  1832. )
  1833. db_session.add(spool)
  1834. await db_session.flush()
  1835. # K-profile is bound to the spool, not to a slot
  1836. db_session.add(
  1837. SpoolKProfile(
  1838. spool_id=spool.id,
  1839. printer_id=printer.id,
  1840. extruder=0,
  1841. nozzle_diameter="0.4",
  1842. k_value=0.025,
  1843. cali_idx=42,
  1844. )
  1845. )
  1846. # NOTE: deliberately no SpoolAssignment — that's the bug condition.
  1847. await db_session.commit()
  1848. mock_client = MagicMock()
  1849. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1850. state = _build_h2d_state(cali_idx=5)
  1851. with (
  1852. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1853. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1854. _patch_async_session_to(db_session),
  1855. ):
  1856. mock_pm.get_client.return_value = mock_client
  1857. mock_pm.get_status.return_value = state
  1858. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1859. # Stage 1b should match the spool by tray_uuid → stored cali_idx=42 wins
  1860. # over live cali_idx=5. Pre-fix this would have been 5 (firmware default).
  1861. mock_client.extrusion_cali_sel.assert_called_once()
  1862. assert mock_client.extrusion_cali_sel.call_args.kwargs["cali_idx"] == 42
  1863. @pytest.mark.asyncio
  1864. @pytest.mark.integration
  1865. async def test_tag_fallback_matches_by_tag_uid_when_uuid_zero(
  1866. self,
  1867. db_session,
  1868. printer_factory,
  1869. ):
  1870. """Stage 1b: when tray_uuid is the zero sentinel but tag_uid is real,
  1871. match by tag_uid. Older firmwares occasionally report a zero tray_uuid
  1872. right after RFID re-read while the tag_uid is already populated."""
  1873. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1874. from backend.app.models.spool import Spool
  1875. from backend.app.models.spool_k_profile import SpoolKProfile
  1876. printer = await printer_factory()
  1877. # Spool indexed by tag_uid, not tray_uuid
  1878. spool = Spool(material="PLA", tag_uid="AABBCC1122334400")
  1879. db_session.add(spool)
  1880. await db_session.flush()
  1881. db_session.add(
  1882. SpoolKProfile(
  1883. spool_id=spool.id,
  1884. printer_id=printer.id,
  1885. extruder=0,
  1886. nozzle_diameter="0.4",
  1887. k_value=0.025,
  1888. cali_idx=99,
  1889. )
  1890. )
  1891. await db_session.commit()
  1892. mock_client = MagicMock()
  1893. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1894. # Build a state where the tray reports a real tag_uid but a zero tray_uuid
  1895. # while still passing is_bambu_tag (tag_uid + tray_info_idx is sufficient).
  1896. state = _build_h2d_state(cali_idx=5)
  1897. state.raw_data["ams"][0]["tray"][0]["tray_uuid"] = "00000000000000000000000000000000"
  1898. with (
  1899. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1900. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1901. _patch_async_session_to(db_session),
  1902. ):
  1903. mock_pm.get_client.return_value = mock_client
  1904. mock_pm.get_status.return_value = state
  1905. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1906. mock_client.extrusion_cali_sel.assert_called_once()
  1907. assert mock_client.extrusion_cali_sel.call_args.kwargs["cali_idx"] == 99
  1908. @pytest.mark.asyncio
  1909. @pytest.mark.integration
  1910. async def test_tag_fallback_skipped_when_zero_sentinels(
  1911. self,
  1912. db_session,
  1913. printer_factory,
  1914. ):
  1915. """Stage 1b: when both tray_uuid and tag_uid are zero sentinels, the
  1916. fallback must not match any spool (would otherwise pick up an
  1917. unrelated spool created with empty/zero tag fields). Falls through
  1918. to Stage 3 live cali_idx as before.
  1919. """
  1920. from backend.app.api.routes.printers import _apply_pa_after_refresh
  1921. from backend.app.models.spool import Spool
  1922. from backend.app.models.spool_k_profile import SpoolKProfile
  1923. printer = await printer_factory()
  1924. # Decoy spool with no tag info — must NOT match
  1925. spool = Spool(material="PLA")
  1926. db_session.add(spool)
  1927. await db_session.flush()
  1928. db_session.add(
  1929. SpoolKProfile(
  1930. spool_id=spool.id,
  1931. printer_id=printer.id,
  1932. extruder=0,
  1933. nozzle_diameter="0.4",
  1934. k_value=0.025,
  1935. cali_idx=42,
  1936. )
  1937. )
  1938. await db_session.commit()
  1939. mock_client = MagicMock()
  1940. mock_client.extrusion_cali_sel = MagicMock(return_value=True)
  1941. state = _build_h2d_state(cali_idx=7)
  1942. # Force both tag fields to the zero sentinels but keep tray_info_idx
  1943. # so is_bambu_tag still passes (preset present)
  1944. state.raw_data["ams"][0]["tray"][0]["tag_uid"] = "0000000000000000"
  1945. state.raw_data["ams"][0]["tray"][0]["tray_uuid"] = "00000000000000000000000000000000"
  1946. with (
  1947. patch("backend.app.api.routes.printers.asyncio.sleep", AsyncMock()),
  1948. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  1949. _patch_async_session_to(db_session),
  1950. ):
  1951. mock_pm.get_client.return_value = mock_client
  1952. mock_pm.get_status.return_value = state
  1953. # is_bambu_tag actually rejects both-zero + only-preset, so the
  1954. # function returns early. We just want to confirm we didn't blow
  1955. # up scanning for a tag-fallback spool.
  1956. await _apply_pa_after_refresh(printer.id, ams_id=0, slot_id=2)
  1957. # is_bambu_tag short-circuits early when both UID and UUID are zero,
  1958. # so no MQTT call should fire and the decoy spool's cali_idx=42 must
  1959. # NOT leak through.
  1960. if mock_client.extrusion_cali_sel.called:
  1961. assert mock_client.extrusion_cali_sel.call_args.kwargs["cali_idx"] != 42
  1962. class TestConfigureAmsSlotPersistsKProfile:
  1963. """Phase 13 P13-T-BE-2: configure_ams_slot persists K-profile to DB.
  1964. Pre-Phase-13 the endpoint sent extrusion_cali_sel via MQTT but never
  1965. recorded the choice in spool_k_profile / spoolman_k_profile, so the next
  1966. RFID re-read had no stored profile to apply.
  1967. """
  1968. @pytest.mark.asyncio
  1969. @pytest.mark.integration
  1970. async def test_writes_spoolman_kprofile_when_spoolman_assigned(
  1971. self,
  1972. async_client: AsyncClient,
  1973. db_session,
  1974. printer_factory,
  1975. ):
  1976. """SpoolmanSlotAssignment present → SpoolmanKProfile row created with cali_idx + k_value + name."""
  1977. from backend.app.models.spoolman_k_profile import SpoolmanKProfile
  1978. from backend.app.models.spoolman_slot_assignment import SpoolmanSlotAssignment
  1979. printer = await printer_factory(model="H2D")
  1980. db_session.add(
  1981. SpoolmanSlotAssignment(
  1982. printer_id=printer.id,
  1983. ams_id=0,
  1984. tray_id=3,
  1985. spoolman_spool_id=216,
  1986. )
  1987. )
  1988. await db_session.commit()
  1989. mock_client = MagicMock()
  1990. mock_client.ams_set_filament_setting.return_value = True
  1991. mock_client.extrusion_cali_sel.return_value = True
  1992. mock_client.extrusion_cali_set.return_value = True
  1993. mock_client.request_status_update.return_value = True
  1994. mock_state = MagicMock()
  1995. mock_state.ams_extruder_map = {"0": 0}
  1996. mock_state.raw_data = {"ams": {"ams": []}}
  1997. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  1998. mock_pm.get_client.return_value = mock_client
  1999. mock_pm.get_status.return_value = mock_state
  2000. response = await async_client.post(
  2001. f"/api/v1/printers/{printer.id}/slots/0/3/configure",
  2002. params={
  2003. "tray_info_idx": "GFL05",
  2004. "tray_type": "PLA",
  2005. "tray_sub_brands": "Bambu PLA Metal",
  2006. "tray_color": "FF8800FF",
  2007. "nozzle_temp_min": 220,
  2008. "nozzle_temp_max": 240,
  2009. "cali_idx": 5,
  2010. "nozzle_diameter": "0.4",
  2011. "k_value": 0.022,
  2012. },
  2013. )
  2014. assert response.status_code == 200
  2015. kp_result = await db_session.execute(select(SpoolmanKProfile).where(SpoolmanKProfile.spoolman_spool_id == 216))
  2016. kp = kp_result.scalar_one_or_none()
  2017. assert kp is not None
  2018. assert kp.cali_idx == 5
  2019. assert kp.k_value == pytest.approx(0.022)
  2020. assert kp.extruder == 0
  2021. assert kp.nozzle_diameter == "0.4"
  2022. assert kp.name == "Bambu PLA Metal"
  2023. @pytest.mark.asyncio
  2024. @pytest.mark.integration
  2025. async def test_writes_spool_kprofile_when_local_assigned(
  2026. self,
  2027. async_client: AsyncClient,
  2028. db_session,
  2029. printer_factory,
  2030. ):
  2031. """Local SpoolAssignment present → SpoolKProfile row created."""
  2032. from backend.app.models.spool import Spool
  2033. from backend.app.models.spool_assignment import SpoolAssignment
  2034. from backend.app.models.spool_k_profile import SpoolKProfile
  2035. printer = await printer_factory(model="H2D")
  2036. spool = Spool(material="PLA", color_name="Red", rgba="FF0000FF")
  2037. db_session.add(spool)
  2038. await db_session.flush()
  2039. db_session.add(
  2040. SpoolAssignment(
  2041. spool_id=spool.id,
  2042. printer_id=printer.id,
  2043. ams_id=0,
  2044. tray_id=3,
  2045. )
  2046. )
  2047. await db_session.commit()
  2048. spool_id = spool.id
  2049. mock_client = MagicMock()
  2050. mock_client.ams_set_filament_setting.return_value = True
  2051. mock_client.extrusion_cali_sel.return_value = True
  2052. mock_client.extrusion_cali_set.return_value = True
  2053. mock_client.request_status_update.return_value = True
  2054. mock_state = MagicMock()
  2055. mock_state.ams_extruder_map = {"0": 0}
  2056. mock_state.raw_data = {"ams": {"ams": []}}
  2057. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  2058. mock_pm.get_client.return_value = mock_client
  2059. mock_pm.get_status.return_value = mock_state
  2060. response = await async_client.post(
  2061. f"/api/v1/printers/{printer.id}/slots/0/3/configure",
  2062. params={
  2063. "tray_info_idx": "PFUSdev01",
  2064. "tray_type": "PLA",
  2065. "tray_sub_brands": "Devil Design PLA",
  2066. "tray_color": "FF0000FF",
  2067. "nozzle_temp_min": 220,
  2068. "nozzle_temp_max": 240,
  2069. "cali_idx": 7,
  2070. "nozzle_diameter": "0.4",
  2071. "k_value": 0.028,
  2072. },
  2073. )
  2074. assert response.status_code == 200
  2075. kp_result = await db_session.execute(select(SpoolKProfile).where(SpoolKProfile.spool_id == spool_id))
  2076. kp = kp_result.scalar_one_or_none()
  2077. assert kp is not None
  2078. assert kp.cali_idx == 7
  2079. assert kp.k_value == pytest.approx(0.028)
  2080. assert kp.extruder == 0
  2081. assert kp.nozzle_diameter == "0.4"
  2082. assert kp.name == "Devil Design PLA"
  2083. @pytest.mark.asyncio
  2084. @pytest.mark.integration
  2085. async def test_no_assignment_no_persist(
  2086. self,
  2087. async_client: AsyncClient,
  2088. db_session,
  2089. printer_factory,
  2090. ):
  2091. """No SpoolAssignment AND no SpoolmanSlotAssignment → no DB write, MQTT still sent."""
  2092. from backend.app.models.spool_k_profile import SpoolKProfile
  2093. from backend.app.models.spoolman_k_profile import SpoolmanKProfile
  2094. printer = await printer_factory(model="H2D")
  2095. mock_client = MagicMock()
  2096. mock_client.ams_set_filament_setting.return_value = True
  2097. mock_client.extrusion_cali_sel.return_value = True
  2098. mock_client.request_status_update.return_value = True
  2099. mock_state = MagicMock()
  2100. mock_state.ams_extruder_map = {"0": 0}
  2101. mock_state.raw_data = {"ams": {"ams": []}}
  2102. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  2103. mock_pm.get_client.return_value = mock_client
  2104. mock_pm.get_status.return_value = mock_state
  2105. response = await async_client.post(
  2106. f"/api/v1/printers/{printer.id}/slots/0/3/configure",
  2107. params={
  2108. "tray_info_idx": "GFL05",
  2109. "tray_type": "PLA",
  2110. "tray_sub_brands": "PLA Basic",
  2111. "tray_color": "FFFFFFFF",
  2112. "nozzle_temp_min": 190,
  2113. "nozzle_temp_max": 230,
  2114. "cali_idx": 5,
  2115. "k_value": 0.020,
  2116. },
  2117. )
  2118. assert response.status_code == 200
  2119. # MQTT sent (was successful), but no DB writes
  2120. mock_client.extrusion_cali_sel.assert_called_once()
  2121. local_count = (await db_session.execute(select(SpoolKProfile))).scalars().all()
  2122. sm_count = (await db_session.execute(select(SpoolmanKProfile))).scalars().all()
  2123. assert len(local_count) == 0
  2124. assert len(sm_count) == 0
  2125. @pytest.mark.asyncio
  2126. @pytest.mark.integration
  2127. async def test_negative_cali_idx_no_persist(
  2128. self,
  2129. async_client: AsyncClient,
  2130. db_session,
  2131. printer_factory,
  2132. ):
  2133. """cali_idx=-1 (no profile selected) → no DB write even when assignment exists."""
  2134. from backend.app.models.spoolman_k_profile import SpoolmanKProfile
  2135. from backend.app.models.spoolman_slot_assignment import SpoolmanSlotAssignment
  2136. printer = await printer_factory(model="H2D")
  2137. db_session.add(
  2138. SpoolmanSlotAssignment(
  2139. printer_id=printer.id,
  2140. ams_id=0,
  2141. tray_id=3,
  2142. spoolman_spool_id=216,
  2143. )
  2144. )
  2145. await db_session.commit()
  2146. mock_client = MagicMock()
  2147. mock_client.ams_set_filament_setting.return_value = True
  2148. mock_client.extrusion_cali_sel.return_value = True
  2149. mock_client.extrusion_cali_set.return_value = True
  2150. mock_client.request_status_update.return_value = True
  2151. mock_state = MagicMock()
  2152. mock_state.ams_extruder_map = {"0": 0}
  2153. mock_state.raw_data = {"ams": {"ams": []}}
  2154. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  2155. mock_pm.get_client.return_value = mock_client
  2156. mock_pm.get_status.return_value = mock_state
  2157. response = await async_client.post(
  2158. f"/api/v1/printers/{printer.id}/slots/0/3/configure",
  2159. params={
  2160. "tray_info_idx": "GFL05",
  2161. "tray_type": "PLA",
  2162. "tray_sub_brands": "PLA Basic",
  2163. "tray_color": "FFFFFFFF",
  2164. "nozzle_temp_min": 190,
  2165. "nozzle_temp_max": 230,
  2166. "cali_idx": -1,
  2167. "k_value": 0.0,
  2168. },
  2169. )
  2170. assert response.status_code == 200
  2171. sm_kps = (
  2172. (await db_session.execute(select(SpoolmanKProfile).where(SpoolmanKProfile.spoolman_spool_id == 216)))
  2173. .scalars()
  2174. .all()
  2175. )
  2176. assert len(sm_kps) == 0 # cali_idx=-1 means "no profile" — don't write
  2177. @pytest.mark.asyncio
  2178. @pytest.mark.integration
  2179. async def test_zero_cali_idx_persists(
  2180. self,
  2181. async_client: AsyncClient,
  2182. db_session,
  2183. printer_factory,
  2184. ):
  2185. """cali_idx=0 is the first valid profile slot (NOT a sentinel for missing)."""
  2186. from backend.app.models.spoolman_k_profile import SpoolmanKProfile
  2187. from backend.app.models.spoolman_slot_assignment import SpoolmanSlotAssignment
  2188. printer = await printer_factory(model="H2D")
  2189. db_session.add(
  2190. SpoolmanSlotAssignment(
  2191. printer_id=printer.id,
  2192. ams_id=0,
  2193. tray_id=3,
  2194. spoolman_spool_id=216,
  2195. )
  2196. )
  2197. await db_session.commit()
  2198. mock_client = MagicMock()
  2199. mock_client.ams_set_filament_setting.return_value = True
  2200. mock_client.extrusion_cali_sel.return_value = True
  2201. mock_client.request_status_update.return_value = True
  2202. mock_state = MagicMock()
  2203. mock_state.ams_extruder_map = {"0": 0}
  2204. mock_state.raw_data = {"ams": {"ams": []}}
  2205. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  2206. mock_pm.get_client.return_value = mock_client
  2207. mock_pm.get_status.return_value = mock_state
  2208. response = await async_client.post(
  2209. f"/api/v1/printers/{printer.id}/slots/0/3/configure",
  2210. params={
  2211. "tray_info_idx": "GFL05",
  2212. "tray_type": "PLA",
  2213. "tray_sub_brands": "PLA Basic",
  2214. "tray_color": "FFFFFFFF",
  2215. "nozzle_temp_min": 190,
  2216. "nozzle_temp_max": 230,
  2217. "cali_idx": 0,
  2218. "k_value": 0.020,
  2219. },
  2220. )
  2221. assert response.status_code == 200
  2222. kp = (
  2223. await db_session.execute(select(SpoolmanKProfile).where(SpoolmanKProfile.spoolman_spool_id == 216))
  2224. ).scalar_one_or_none()
  2225. assert kp is not None
  2226. assert kp.cali_idx == 0 # explicitly testing 0 is valid
  2227. @pytest.mark.asyncio
  2228. @pytest.mark.integration
  2229. async def test_upsert_idempotent(
  2230. self,
  2231. async_client: AsyncClient,
  2232. db_session,
  2233. printer_factory,
  2234. ):
  2235. """Repeated POSTs update the same row (UNIQUE on spool_id+printer+extruder+nozzle_diameter)."""
  2236. from backend.app.models.spoolman_k_profile import SpoolmanKProfile
  2237. from backend.app.models.spoolman_slot_assignment import SpoolmanSlotAssignment
  2238. printer = await printer_factory(model="H2D")
  2239. db_session.add(
  2240. SpoolmanSlotAssignment(
  2241. printer_id=printer.id,
  2242. ams_id=0,
  2243. tray_id=3,
  2244. spoolman_spool_id=216,
  2245. )
  2246. )
  2247. await db_session.commit()
  2248. mock_client = MagicMock()
  2249. mock_client.ams_set_filament_setting.return_value = True
  2250. mock_client.extrusion_cali_sel.return_value = True
  2251. mock_client.request_status_update.return_value = True
  2252. mock_state = MagicMock()
  2253. mock_state.ams_extruder_map = {"0": 0}
  2254. mock_state.raw_data = {"ams": {"ams": []}}
  2255. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  2256. mock_pm.get_client.return_value = mock_client
  2257. mock_pm.get_status.return_value = mock_state
  2258. # First call with cali_idx=5
  2259. await async_client.post(
  2260. f"/api/v1/printers/{printer.id}/slots/0/3/configure",
  2261. params={
  2262. "tray_info_idx": "GFL05",
  2263. "tray_type": "PLA",
  2264. "tray_sub_brands": "PLA Basic",
  2265. "tray_color": "FFFFFFFF",
  2266. "nozzle_temp_min": 190,
  2267. "nozzle_temp_max": 230,
  2268. "cali_idx": 5,
  2269. "k_value": 0.020,
  2270. },
  2271. )
  2272. # Second call with cali_idx=10 (same slot/spool/extruder/nozzle)
  2273. await async_client.post(
  2274. f"/api/v1/printers/{printer.id}/slots/0/3/configure",
  2275. params={
  2276. "tray_info_idx": "GFL05",
  2277. "tray_type": "PLA",
  2278. "tray_sub_brands": "PLA Matte",
  2279. "tray_color": "FFFFFFFF",
  2280. "nozzle_temp_min": 190,
  2281. "nozzle_temp_max": 230,
  2282. "cali_idx": 10,
  2283. "k_value": 0.025,
  2284. },
  2285. )
  2286. # Should be exactly ONE row (updated), not two
  2287. kps = (
  2288. (await db_session.execute(select(SpoolmanKProfile).where(SpoolmanKProfile.spoolman_spool_id == 216)))
  2289. .scalars()
  2290. .all()
  2291. )
  2292. assert len(kps) == 1
  2293. assert kps[0].cali_idx == 10 # updated to most recent
  2294. assert kps[0].name == "PLA Matte"
  2295. @pytest.mark.asyncio
  2296. @pytest.mark.integration
  2297. async def test_external_slot_extruder_inversion(
  2298. self,
  2299. async_client: AsyncClient,
  2300. db_session,
  2301. printer_factory,
  2302. ):
  2303. """ams_id=255 + tray_id=0 → kp.extruder=1 (ext-L); tray_id=1 → extruder=0 (ext-R)."""
  2304. from backend.app.models.spool import Spool
  2305. from backend.app.models.spool_assignment import SpoolAssignment
  2306. from backend.app.models.spool_k_profile import SpoolKProfile
  2307. printer = await printer_factory(model="H2D")
  2308. spool = Spool(material="PLA")
  2309. db_session.add(spool)
  2310. await db_session.flush()
  2311. # Note: SpoolmanSlotAssignment can't store ams_id=255 with tray_id=1
  2312. # under the ck_tray_id_range constraint (0-3 valid). External-slot
  2313. # K-profile persistence is therefore tested via local SpoolAssignment.
  2314. db_session.add(
  2315. SpoolAssignment(
  2316. spool_id=spool.id,
  2317. printer_id=printer.id,
  2318. ams_id=255,
  2319. tray_id=0,
  2320. )
  2321. )
  2322. await db_session.commit()
  2323. spool_id = spool.id
  2324. mock_client = MagicMock()
  2325. mock_client.ams_set_filament_setting.return_value = True
  2326. mock_client.extrusion_cali_sel.return_value = True
  2327. mock_client.request_status_update.return_value = True
  2328. mock_state = MagicMock()
  2329. mock_state.ams_extruder_map = {"0": 0} # truthy so external-inversion path runs
  2330. mock_state.raw_data = {"ams": {"ams": []}}
  2331. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  2332. mock_pm.get_client.return_value = mock_client
  2333. mock_pm.get_status.return_value = mock_state
  2334. response = await async_client.post(
  2335. f"/api/v1/printers/{printer.id}/slots/255/0/configure",
  2336. params={
  2337. "tray_info_idx": "GFL05",
  2338. "tray_type": "PLA",
  2339. "tray_sub_brands": "PLA Basic",
  2340. "tray_color": "FFFFFFFF",
  2341. "nozzle_temp_min": 190,
  2342. "nozzle_temp_max": 230,
  2343. "cali_idx": 5,
  2344. "k_value": 0.020,
  2345. },
  2346. )
  2347. assert response.status_code == 200
  2348. kp = (
  2349. await db_session.execute(select(SpoolKProfile).where(SpoolKProfile.spool_id == spool_id))
  2350. ).scalar_one_or_none()
  2351. assert kp is not None
  2352. # tray_id=0 → extruder = 1 - 0 = 1
  2353. assert kp.extruder == 1
  2354. @pytest.mark.asyncio
  2355. @pytest.mark.integration
  2356. async def test_dual_nozzle_extruder_persists(
  2357. self,
  2358. async_client: AsyncClient,
  2359. db_session,
  2360. printer_factory,
  2361. ):
  2362. """ams_extruder_map with extruder=1 → kp.extruder=1 persisted correctly."""
  2363. from backend.app.models.spool import Spool
  2364. from backend.app.models.spool_assignment import SpoolAssignment
  2365. from backend.app.models.spool_k_profile import SpoolKProfile
  2366. printer = await printer_factory(model="H2D")
  2367. spool = Spool(material="PLA")
  2368. db_session.add(spool)
  2369. await db_session.flush()
  2370. db_session.add(
  2371. SpoolAssignment(
  2372. spool_id=spool.id,
  2373. printer_id=printer.id,
  2374. ams_id=2,
  2375. tray_id=3,
  2376. )
  2377. )
  2378. await db_session.commit()
  2379. spool_id = spool.id
  2380. mock_client = MagicMock()
  2381. mock_client.ams_set_filament_setting.return_value = True
  2382. mock_client.extrusion_cali_sel.return_value = True
  2383. mock_client.request_status_update.return_value = True
  2384. mock_state = MagicMock()
  2385. mock_state.ams_extruder_map = {"2": 1} # AMS 2 is on extruder 1
  2386. mock_state.raw_data = {"ams": {"ams": []}}
  2387. with patch("backend.app.api.routes.printers.printer_manager") as mock_pm:
  2388. mock_pm.get_client.return_value = mock_client
  2389. mock_pm.get_status.return_value = mock_state
  2390. response = await async_client.post(
  2391. f"/api/v1/printers/{printer.id}/slots/2/3/configure",
  2392. params={
  2393. "tray_info_idx": "GFL05",
  2394. "tray_type": "PLA",
  2395. "tray_sub_brands": "PLA Basic",
  2396. "tray_color": "FFFFFFFF",
  2397. "nozzle_temp_min": 190,
  2398. "nozzle_temp_max": 230,
  2399. "cali_idx": 5,
  2400. "k_value": 0.020,
  2401. },
  2402. )
  2403. assert response.status_code == 200
  2404. kp = (
  2405. await db_session.execute(select(SpoolKProfile).where(SpoolKProfile.spool_id == spool_id))
  2406. ).scalar_one_or_none()
  2407. assert kp is not None
  2408. assert kp.extruder == 1
  2409. @pytest.mark.asyncio
  2410. @pytest.mark.integration
  2411. async def test_db_error_does_not_fail_endpoint(
  2412. self,
  2413. async_client: AsyncClient,
  2414. db_session,
  2415. printer_factory,
  2416. ):
  2417. """DB errors during K-profile persistence are best-effort — endpoint still returns 200.
  2418. Verifies the try/except wrap added in P13-3b: if DB upsert fails (e.g.
  2419. because the schema is out of sync, a constraint violation, or any
  2420. other transient error), the MQTT command was already sent successfully
  2421. so we shouldn't return 500 to the user. The error is logged and the
  2422. endpoint returns success.
  2423. """
  2424. from backend.app.models.spoolman_slot_assignment import SpoolmanSlotAssignment
  2425. printer = await printer_factory(model="H2D")
  2426. db_session.add(
  2427. SpoolmanSlotAssignment(
  2428. printer_id=printer.id,
  2429. ams_id=0,
  2430. tray_id=3,
  2431. spoolman_spool_id=216,
  2432. )
  2433. )
  2434. await db_session.commit()
  2435. mock_client = MagicMock()
  2436. mock_client.ams_set_filament_setting.return_value = True
  2437. mock_client.extrusion_cali_sel.return_value = True
  2438. mock_client.request_status_update.return_value = True
  2439. mock_state = MagicMock()
  2440. mock_state.ams_extruder_map = {"0": 0}
  2441. mock_state.raw_data = {"ams": {"ams": []}}
  2442. # Force the K-profile persistence path to fail by patching the
  2443. # SpoolmanKProfile model class with a sentinel that raises when
  2444. # instantiated. The MQTT call has already happened by then, so the
  2445. # endpoint must catch and log without returning 500.
  2446. with (
  2447. patch("backend.app.api.routes.printers.printer_manager") as mock_pm,
  2448. patch(
  2449. "backend.app.models.spoolman_k_profile.SpoolmanKProfile",
  2450. side_effect=RuntimeError("Simulated DB error"),
  2451. ),
  2452. ):
  2453. mock_pm.get_client.return_value = mock_client
  2454. mock_pm.get_status.return_value = mock_state
  2455. response = await async_client.post(
  2456. f"/api/v1/printers/{printer.id}/slots/0/3/configure",
  2457. params={
  2458. "tray_info_idx": "GFL05",
  2459. "tray_type": "PLA",
  2460. "tray_sub_brands": "PLA Basic",
  2461. "tray_color": "FFFFFFFF",
  2462. "nozzle_temp_min": 190,
  2463. "nozzle_temp_max": 230,
  2464. "cali_idx": 5,
  2465. "k_value": 0.020,
  2466. },
  2467. )
  2468. # Endpoint returns success — MQTT was sent, K-profile failed silently
  2469. assert response.status_code == 200
  2470. # MQTT was indeed called
  2471. mock_client.extrusion_cali_sel.assert_called_once()