test_usage_tracker.py 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766
  1. """Unit tests for the filament usage tracker.
  2. Tests 3MF-primary tracking (Path 1) and AMS remain% delta fallback
  3. (Path 2) for spools not covered by 3MF data.
  4. """
  5. from datetime import datetime, timezone
  6. from unittest.mock import AsyncMock, MagicMock, patch
  7. import pytest
  8. from backend.app.services.usage_tracker import (
  9. PrintSession,
  10. _active_sessions,
  11. _track_from_3mf,
  12. on_print_complete,
  13. on_print_start,
  14. )
  15. def _make_spool(*, id=1, label_weight=1000, weight_used=0, tag_uid=None, tray_uuid=None):
  16. """Create a mock Spool object."""
  17. spool = MagicMock()
  18. spool.id = id
  19. spool.label_weight = label_weight
  20. spool.weight_used = weight_used
  21. spool.tag_uid = tag_uid
  22. spool.tray_uuid = tray_uuid
  23. spool.last_used = None
  24. spool.cost_per_kg = None
  25. spool.material = "PLA"
  26. return spool
  27. def _make_assignment(*, spool_id=1, printer_id=1, ams_id=0, tray_id=0, created_at=None):
  28. """Create a mock SpoolAssignment object."""
  29. assignment = MagicMock()
  30. assignment.spool_id = spool_id
  31. assignment.printer_id = printer_id
  32. assignment.ams_id = ams_id
  33. assignment.tray_id = tray_id
  34. assignment.created_at = created_at or datetime.now(timezone.utc)
  35. return assignment
  36. def _make_printer_state(ams_data, progress=0, layer_num=0, tray_now=255):
  37. """Create a mock printer state with AMS data."""
  38. state = MagicMock()
  39. state.raw_data = {"ams": ams_data}
  40. state.progress = progress
  41. state.layer_num = layer_num
  42. state.tray_now = tray_now
  43. return state
  44. def _make_printer_manager(state=None):
  45. """Create a mock printer manager."""
  46. pm = MagicMock()
  47. pm.get_status.return_value = state
  48. return pm
  49. class TestOnPrintStart:
  50. """Tests for on_print_start — capturing AMS remain%."""
  51. @pytest.fixture(autouse=True)
  52. def _clear_sessions(self):
  53. _active_sessions.clear()
  54. yield
  55. _active_sessions.clear()
  56. @pytest.mark.asyncio
  57. async def test_creates_session_with_valid_remain(self):
  58. """Session created with remain% data for trays reporting 0-100."""
  59. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 80}]}]
  60. pm = _make_printer_manager(_make_printer_state(ams_data))
  61. await on_print_start(1, {"subtask_name": "test_print"}, pm)
  62. assert 1 in _active_sessions
  63. session = _active_sessions[1]
  64. assert session.print_name == "test_print"
  65. assert session.tray_remain_start == {(0, 0): 80}
  66. @pytest.mark.asyncio
  67. async def test_creates_session_even_without_valid_remain(self):
  68. """Session still created when remain=-1 (for 3MF fallback path)."""
  69. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": -1}]}]
  70. pm = _make_printer_manager(_make_printer_state(ams_data))
  71. await on_print_start(1, {"subtask_name": "test_print"}, pm)
  72. assert 1 in _active_sessions
  73. session = _active_sessions[1]
  74. assert session.tray_remain_start == {} # Empty, no valid remain
  75. @pytest.mark.asyncio
  76. async def test_skips_without_ams_data(self):
  77. """No session created when no AMS data available."""
  78. state = MagicMock()
  79. state.raw_data = {"ams": []}
  80. pm = _make_printer_manager(state)
  81. await on_print_start(1, {"subtask_name": "test"}, pm)
  82. assert 1 not in _active_sessions
  83. class TestOnPrintCompleteAMSDelta:
  84. """Tests for Path 1: AMS remain% delta tracking."""
  85. @pytest.fixture(autouse=True)
  86. def _clear_sessions(self):
  87. _active_sessions.clear()
  88. yield
  89. _active_sessions.clear()
  90. @pytest.fixture(autouse=True)
  91. def _mock_get_setting(self):
  92. with patch(
  93. "backend.app.api.routes.settings.get_setting",
  94. new_callable=AsyncMock,
  95. return_value=None,
  96. ):
  97. yield
  98. @pytest.mark.asyncio
  99. async def test_computes_delta_and_updates_spool(self):
  100. """Spool weight_used updated by remain% delta * label_weight."""
  101. # Set up session with start remain = 80%
  102. _active_sessions[1] = PrintSession(
  103. printer_id=1,
  104. print_name="test",
  105. started_at=datetime.now(timezone.utc),
  106. tray_remain_start={(0, 0): 80},
  107. )
  108. # Current remain = 70% → 10% consumed → 100g on 1000g spool
  109. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 70}]}]
  110. pm = _make_printer_manager(_make_printer_state(ams_data))
  111. spool = _make_spool(label_weight=1000, weight_used=50)
  112. assignment = _make_assignment()
  113. db = AsyncMock()
  114. # First 2 executes → _find_3mf_by_filename (library + archive search, uses scalars().all()),
  115. # then assignment, then spool for the AMS fallback path
  116. db.execute = AsyncMock(
  117. side_effect=[
  118. MagicMock(), # _find_3mf_by_filename: library search
  119. MagicMock(), # _find_3mf_by_filename: archive search
  120. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  121. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  122. ]
  123. )
  124. results = await on_print_complete(1, {"status": "completed"}, pm, db)
  125. assert len(results) == 1
  126. assert results[0]["weight_used"] == 100.0
  127. assert results[0]["percent_used"] == 10
  128. # weight_used should be old (50) + delta (100)
  129. assert spool.weight_used == 150.0
  130. db.commit.assert_called_once()
  131. @pytest.mark.asyncio
  132. async def test_skips_negative_delta(self):
  133. """No tracking when remain increased (spool refilled)."""
  134. _active_sessions[1] = PrintSession(
  135. printer_id=1,
  136. print_name="test",
  137. started_at=datetime.now(timezone.utc),
  138. tray_remain_start={(0, 0): 50},
  139. )
  140. # Remain went UP: 50 → 80 (refilled)
  141. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 80}]}]
  142. pm = _make_printer_manager(_make_printer_state(ams_data))
  143. db = AsyncMock()
  144. results = await on_print_complete(1, {"status": "completed"}, pm, db)
  145. assert results == []
  146. db.commit.assert_not_called()
  147. @pytest.mark.asyncio
  148. async def test_no_session_falls_through_to_3mf(self):
  149. """When no session exists, AMS delta path skipped (3MF may still run)."""
  150. pm = _make_printer_manager()
  151. db = AsyncMock()
  152. results = await on_print_complete(1, {"status": "completed"}, pm, db)
  153. assert results == []
  154. @pytest.mark.asyncio
  155. async def test_skips_fallback_for_trays_outside_print_mapping(self):
  156. """#1269: swapping a spool in an UNUSED slot mid-print must NOT charge the old spool.
  157. Reproduces maugsburger's report: single-color print on AMS0-T3
  158. (ams_mapping=[3]). User swaps spools in T1 and T2 during the print —
  159. those slots report remain=0 at completion (new spool with no tag).
  160. The fallback must skip T1 and T2 because they were never in the
  161. print's tray mapping or runtime tray_change_log.
  162. """
  163. _active_sessions[1] = PrintSession(
  164. printer_id=1,
  165. print_name="splitter",
  166. started_at=datetime.now(timezone.utc),
  167. tray_remain_start={(0, 1): 100, (0, 2): 17, (0, 3): 100},
  168. tray_now_at_start=3,
  169. ams_mapping=[3],
  170. )
  171. # User swapped T1 and T2 mid-print → both report remain=0 now.
  172. # T3 was actually used but it's also at 0 now. Without the fix the
  173. # fallback would charge the originally-assigned spools at T1 and T2.
  174. ams_data = [
  175. {
  176. "id": 0,
  177. "tray": [
  178. {"id": 1, "remain": 0},
  179. {"id": 2, "remain": 0},
  180. {"id": 3, "remain": 0},
  181. ],
  182. }
  183. ]
  184. state = _make_printer_state(ams_data, tray_now=3)
  185. state.tray_change_log = [(3, 0)] # only T3 was loaded during the print
  186. pm = _make_printer_manager(state)
  187. # Only T3 should reach the spool lookup; T1 and T2 must be filtered
  188. # out before any DB query is issued for them.
  189. t3_spool = _make_spool(id=8, label_weight=1000, weight_used=0)
  190. t3_assignment = _make_assignment(spool_id=8, ams_id=0, tray_id=3)
  191. db = AsyncMock()
  192. db.execute = AsyncMock(
  193. side_effect=[
  194. MagicMock(), # _find_3mf_by_filename: library search
  195. MagicMock(), # _find_3mf_by_filename: archive search
  196. MagicMock(scalar_one_or_none=MagicMock(return_value=t3_assignment)),
  197. MagicMock(scalar_one_or_none=MagicMock(return_value=t3_spool)),
  198. ]
  199. )
  200. results = await on_print_complete(1, {"status": "completed"}, pm, db)
  201. # Only T3 should be charged. T1 (spool 27 in the report) and T2
  202. # (spool 24) must NOT appear in the results.
  203. assert len(results) == 1
  204. assert results[0]["ams_id"] == 0
  205. assert results[0]["tray_id"] == 3
  206. class TestTrackFrom3MF:
  207. """Tests for Path 2: 3MF per-filament fallback tracking."""
  208. @pytest.mark.asyncio
  209. async def test_updates_non_bl_spool_from_3mf(self):
  210. """Non-BL spool gets weight_used from 3MF used_g for completed print."""
  211. spool = _make_spool(id=5, label_weight=1000, weight_used=100)
  212. assignment = _make_assignment(spool_id=5)
  213. archive = MagicMock()
  214. archive.file_path = "archives/test.3mf"
  215. db = AsyncMock()
  216. # archive, queue_item(None), assignment, spool
  217. db.execute = AsyncMock(
  218. side_effect=[
  219. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  220. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  221. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  222. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  223. ]
  224. )
  225. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  226. filament_usage = [{"slot_id": 1, "used_g": 25.5, "type": "PLA", "color": "#FF0000"}]
  227. with (
  228. patch("backend.app.core.config.settings") as mock_settings,
  229. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  230. ):
  231. mock_path = MagicMock()
  232. mock_path.exists.return_value = True
  233. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  234. results = await _track_from_3mf(
  235. printer_id=1,
  236. archive_id=10,
  237. status="completed",
  238. print_name="test_print",
  239. handled_trays=set(),
  240. printer_manager=pm,
  241. db=db,
  242. )
  243. assert len(results) == 1
  244. assert results[0]["spool_id"] == 5
  245. assert results[0]["weight_used"] == 25.5
  246. # weight_used = old (100) + 3MF (25.5)
  247. assert spool.weight_used == 125.5
  248. @pytest.mark.asyncio
  249. async def test_scales_by_progress_for_failed_print(self):
  250. """Failed print scales 3MF estimate by progress percentage."""
  251. spool = _make_spool(id=1, label_weight=1000, weight_used=0)
  252. assignment = _make_assignment()
  253. archive = MagicMock()
  254. archive.file_path = "archives/test.3mf"
  255. db = AsyncMock()
  256. # archive, queue_item(None), assignment, spool
  257. db.execute = AsyncMock(
  258. side_effect=[
  259. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  260. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  261. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  262. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  263. ]
  264. )
  265. # Print failed at 50% progress → 50g consumed from 100g estimate
  266. pm = _make_printer_manager(_make_printer_state([], progress=50, tray_now=0))
  267. filament_usage = [{"slot_id": 1, "used_g": 100.0, "type": "PLA", "color": ""}]
  268. with (
  269. patch("backend.app.core.config.settings") as mock_settings,
  270. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  271. ):
  272. mock_path = MagicMock()
  273. mock_path.exists.return_value = True
  274. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  275. results = await _track_from_3mf(
  276. printer_id=1,
  277. archive_id=10,
  278. status="failed",
  279. print_name="test",
  280. handled_trays=set(),
  281. printer_manager=pm,
  282. db=db,
  283. )
  284. assert len(results) == 1
  285. assert results[0]["weight_used"] == 50.0
  286. assert spool.weight_used == 50.0
  287. @pytest.mark.asyncio
  288. async def test_tracks_bl_spools_via_3mf(self):
  289. """BL spools (with tag_uid) ARE now tracked via 3MF (unified tracking)."""
  290. spool = _make_spool(tag_uid="ABCD1234", tray_uuid="A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4")
  291. assignment = _make_assignment()
  292. archive = MagicMock()
  293. archive.file_path = "archives/test.3mf"
  294. db = AsyncMock()
  295. # archive, queue_item(None), assignment, spool
  296. db.execute = AsyncMock(
  297. side_effect=[
  298. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  299. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  300. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  301. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  302. ]
  303. )
  304. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  305. filament_usage = [{"slot_id": 1, "used_g": 50.0, "type": "PLA", "color": ""}]
  306. with (
  307. patch("backend.app.core.config.settings") as mock_settings,
  308. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  309. ):
  310. mock_path = MagicMock()
  311. mock_path.exists.return_value = True
  312. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  313. results = await _track_from_3mf(
  314. printer_id=1,
  315. archive_id=10,
  316. status="completed",
  317. print_name="test",
  318. handled_trays=set(),
  319. printer_manager=pm,
  320. db=db,
  321. )
  322. assert len(results) == 1
  323. assert results[0]["spool_id"] == 1
  324. assert results[0]["weight_used"] == 50.0
  325. @pytest.mark.asyncio
  326. async def test_skips_already_handled_trays(self):
  327. """Trays handled by AMS remain% delta are not double-tracked via 3MF."""
  328. archive = MagicMock()
  329. archive.file_path = "archives/test.3mf"
  330. db = AsyncMock()
  331. # archive, queue_item(None)
  332. db.execute = AsyncMock(
  333. side_effect=[
  334. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  335. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  336. ]
  337. )
  338. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  339. filament_usage = [{"slot_id": 1, "used_g": 50.0, "type": "PLA", "color": ""}]
  340. with (
  341. patch("backend.app.core.config.settings") as mock_settings,
  342. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  343. ):
  344. mock_path = MagicMock()
  345. mock_path.exists.return_value = True
  346. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  347. results = await _track_from_3mf(
  348. printer_id=1,
  349. archive_id=10,
  350. status="completed",
  351. print_name="test",
  352. handled_trays={(0, 0)}, # slot_id=1 → ams_id=0, tray_id=0
  353. printer_manager=pm,
  354. db=db,
  355. )
  356. assert results == []
  357. @pytest.mark.asyncio
  358. async def test_slot_to_tray_mapping(self):
  359. """3MF slot_id maps correctly to (ams_id, tray_id) via tray_now."""
  360. # tray_now=4 → ams_id=1, tray_id=0 (single filament uses tray_now)
  361. spool = _make_spool(id=9)
  362. assignment = _make_assignment(spool_id=9, ams_id=1, tray_id=0)
  363. archive = MagicMock()
  364. archive.file_path = "archives/test.3mf"
  365. db = AsyncMock()
  366. # archive, queue_item(None), assignment, spool
  367. db.execute = AsyncMock(
  368. side_effect=[
  369. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  370. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  371. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  372. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  373. ]
  374. )
  375. pm = _make_printer_manager(_make_printer_state([], tray_now=4))
  376. filament_usage = [{"slot_id": 5, "used_g": 30.0, "type": "PETG", "color": ""}]
  377. with (
  378. patch("backend.app.core.config.settings") as mock_settings,
  379. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  380. ):
  381. mock_path = MagicMock()
  382. mock_path.exists.return_value = True
  383. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  384. results = await _track_from_3mf(
  385. printer_id=1,
  386. archive_id=10,
  387. status="completed",
  388. print_name="test",
  389. handled_trays=set(),
  390. printer_manager=pm,
  391. db=db,
  392. )
  393. assert len(results) == 1
  394. assert results[0]["ams_id"] == 1
  395. assert results[0]["tray_id"] == 0
  396. class TestSpoolAssignmentSnapshot:
  397. """Tests for spool assignment snapshotting at print start (#459).
  398. When a spool runs empty mid-print, on_ams_change deletes the SpoolAssignment.
  399. The snapshot captured at print start ensures usage is still attributed correctly.
  400. """
  401. @pytest.fixture(autouse=True)
  402. def _clear_sessions(self):
  403. _active_sessions.clear()
  404. yield
  405. _active_sessions.clear()
  406. @pytest.fixture(autouse=True)
  407. def _mock_get_setting(self):
  408. with patch(
  409. "backend.app.api.routes.settings.get_setting",
  410. new_callable=AsyncMock,
  411. return_value=None,
  412. ):
  413. yield
  414. @pytest.mark.asyncio
  415. async def test_on_print_start_snapshots_assignments_with_db(self):
  416. """on_print_start captures spool assignments when db is provided."""
  417. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 80}, {"id": 1, "remain": 60}]}]
  418. pm = _make_printer_manager(_make_printer_state(ams_data, tray_now=0))
  419. assignment_0 = _make_assignment(spool_id=10, printer_id=1, ams_id=0, tray_id=0)
  420. assignment_1 = _make_assignment(spool_id=20, printer_id=1, ams_id=0, tray_id=1)
  421. db = AsyncMock()
  422. scalars_mock = MagicMock()
  423. scalars_mock.all.return_value = [assignment_0, assignment_1]
  424. result_mock = MagicMock()
  425. result_mock.scalars.return_value = scalars_mock
  426. db.execute = AsyncMock(return_value=result_mock)
  427. await on_print_start(1, {"subtask_name": "Benchy"}, pm, db=db)
  428. session = _active_sessions[1]
  429. assert session.spool_assignments == {(0, 0): 10, (0, 1): 20}
  430. @pytest.mark.asyncio
  431. async def test_on_print_start_empty_snapshot_without_db(self):
  432. """on_print_start creates empty snapshot when no db provided."""
  433. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 80}]}]
  434. pm = _make_printer_manager(_make_printer_state(ams_data, tray_now=0))
  435. await on_print_start(1, {"subtask_name": "Benchy"}, pm)
  436. session = _active_sessions[1]
  437. assert session.spool_assignments == {}
  438. @pytest.mark.asyncio
  439. async def test_3mf_uses_snapshot_instead_of_live_query(self):
  440. """_track_from_3mf uses snapshot spool_id without querying SpoolAssignment."""
  441. spool = _make_spool(id=42, label_weight=1000)
  442. archive = MagicMock()
  443. archive.file_path = "archives/test.3mf"
  444. # db: archive, queue_item(None), spool — NO assignment query needed
  445. db = AsyncMock()
  446. db.execute = AsyncMock(
  447. side_effect=[
  448. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  449. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  450. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  451. ]
  452. )
  453. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  454. filament_usage = [{"slot_id": 1, "used_g": 15.0, "type": "PLA", "color": "#FF0000"}]
  455. with (
  456. patch("backend.app.core.config.settings") as mock_settings,
  457. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  458. ):
  459. mock_path = MagicMock()
  460. mock_path.exists.return_value = True
  461. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  462. results = await _track_from_3mf(
  463. printer_id=1,
  464. archive_id=10,
  465. status="completed",
  466. print_name="Test",
  467. handled_trays=set(),
  468. printer_manager=pm,
  469. db=db,
  470. spool_assignments={(0, 0): 42},
  471. )
  472. assert len(results) == 1
  473. assert results[0]["spool_id"] == 42
  474. assert results[0]["weight_used"] == 15.0
  475. @pytest.mark.asyncio
  476. async def test_3mf_falls_back_to_live_query_without_snapshot(self):
  477. """_track_from_3mf queries SpoolAssignment when no snapshot exists."""
  478. spool = _make_spool(id=5, label_weight=1000)
  479. assignment = _make_assignment(spool_id=5)
  480. archive = MagicMock()
  481. archive.file_path = "archives/test.3mf"
  482. # db: archive, queue_item(None), assignment, spool
  483. db = AsyncMock()
  484. db.execute = AsyncMock(
  485. side_effect=[
  486. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  487. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  488. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  489. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  490. ]
  491. )
  492. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  493. filament_usage = [{"slot_id": 1, "used_g": 10.0, "type": "PLA", "color": "#FF0000"}]
  494. with (
  495. patch("backend.app.core.config.settings") as mock_settings,
  496. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  497. ):
  498. mock_path = MagicMock()
  499. mock_path.exists.return_value = True
  500. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  501. results = await _track_from_3mf(
  502. printer_id=1,
  503. archive_id=10,
  504. status="completed",
  505. print_name="Test",
  506. handled_trays=set(),
  507. printer_manager=pm,
  508. db=db,
  509. spool_assignments=None,
  510. )
  511. assert len(results) == 1
  512. assert results[0]["spool_id"] == 5
  513. @pytest.mark.asyncio
  514. async def test_ams_delta_uses_snapshot_over_live_query(self):
  515. """AMS remain% fallback uses snapshot spool_id instead of live query."""
  516. spool = _make_spool(id=77, label_weight=1000)
  517. _active_sessions[1] = PrintSession(
  518. printer_id=1,
  519. print_name="Benchy",
  520. started_at=datetime.now(timezone.utc),
  521. tray_remain_start={(0, 0): 80},
  522. spool_assignments={(0, 0): 77},
  523. )
  524. # Current remain = 70% → 10% delta → 100g
  525. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 70}]}]
  526. pm = _make_printer_manager(_make_printer_state(ams_data))
  527. # First 2 executes → _find_3mf_by_filename (library + archive search),
  528. # then live assignment check (returns None), then spool lookup by snapshot spool_id
  529. db = AsyncMock()
  530. db.execute = AsyncMock(
  531. side_effect=[
  532. MagicMock(), # _find_3mf_by_filename: library search
  533. MagicMock(), # _find_3mf_by_filename: archive search
  534. MagicMock(scalar_one_or_none=MagicMock(return_value=None)), # live assignment
  535. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  536. ]
  537. )
  538. results = await on_print_complete(
  539. printer_id=1,
  540. data={"status": "completed"},
  541. printer_manager=pm,
  542. db=db,
  543. archive_id=None,
  544. )
  545. assert len(results) == 1
  546. assert results[0]["spool_id"] == 77
  547. assert results[0]["weight_used"] == 100.0
  548. @pytest.mark.asyncio
  549. async def test_ams_delta_falls_back_to_live_query_without_snapshot(self):
  550. """AMS remain% fallback queries SpoolAssignment when snapshot is empty."""
  551. spool = _make_spool(id=33, label_weight=1000)
  552. assignment = _make_assignment(spool_id=33)
  553. _active_sessions[1] = PrintSession(
  554. printer_id=1,
  555. print_name="Benchy",
  556. started_at=datetime.now(timezone.utc),
  557. tray_remain_start={(0, 0): 80},
  558. spool_assignments={}, # Empty snapshot (pre-upgrade session)
  559. )
  560. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 70}]}]
  561. pm = _make_printer_manager(_make_printer_state(ams_data))
  562. # First 2 executes → _find_3mf_by_filename (library + archive search),
  563. # then assignment and spool for the AMS fallback path
  564. db = AsyncMock()
  565. db.execute = AsyncMock(
  566. side_effect=[
  567. MagicMock(), # _find_3mf_by_filename: library search
  568. MagicMock(), # _find_3mf_by_filename: archive search
  569. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  570. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  571. ]
  572. )
  573. results = await on_print_complete(
  574. printer_id=1,
  575. data={"status": "completed"},
  576. printer_manager=pm,
  577. db=db,
  578. archive_id=None,
  579. )
  580. assert len(results) == 1
  581. assert results[0]["spool_id"] == 33
  582. @pytest.mark.asyncio
  583. async def test_snapshot_survives_mid_print_unlink(self):
  584. """Core bug scenario: snapshot provides spool_id after mid-print unlink.
  585. Simulates the #459 scenario: spool runs empty mid-print, on_ams_change
  586. deletes the SpoolAssignment, but the snapshot from print start still
  587. has the spool_id so usage is correctly attributed at print completion.
  588. """
  589. spool = _make_spool(id=8, label_weight=1000, weight_used=50)
  590. archive = MagicMock()
  591. archive.file_path = "archives/big_print.3mf"
  592. # Session was created at print start WITH snapshot
  593. _active_sessions[1] = PrintSession(
  594. printer_id=1,
  595. print_name="Big Print",
  596. started_at=datetime.now(timezone.utc),
  597. tray_remain_start={(0, 0): 90},
  598. spool_assignments={(0, 0): 8}, # Snapshot from print start
  599. )
  600. pm = _make_printer_manager(
  601. _make_printer_state(
  602. [{"id": 0, "tray": [{"id": 0, "remain": 75}]}],
  603. tray_now=0,
  604. )
  605. )
  606. filament_usage = [{"slot_id": 1, "used_g": 14.2, "type": "PLA", "color": "#FF0000"}]
  607. # db: archive, queue_item(None), live assignment(None), spool,
  608. # then cost aggregation queries
  609. # NOTE: No assignment in db — it was deleted by on_ams_change mid-print!
  610. db = AsyncMock()
  611. db.execute = AsyncMock(
  612. side_effect=[
  613. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  614. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  615. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  616. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  617. # Cost aggregation: sum query (uses .scalar()), archive lookup
  618. MagicMock(scalar=MagicMock(return_value=0)),
  619. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  620. ]
  621. )
  622. with (
  623. patch("backend.app.core.config.settings") as mock_settings,
  624. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  625. ):
  626. mock_path = MagicMock()
  627. mock_path.exists.return_value = True
  628. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  629. results = await on_print_complete(
  630. printer_id=1,
  631. data={"status": "completed"},
  632. printer_manager=pm,
  633. db=db,
  634. archive_id=100,
  635. )
  636. # Usage should be tracked despite assignment being deleted mid-print
  637. assert len(results) >= 1
  638. assert results[0]["spool_id"] == 8
  639. assert results[0]["weight_used"] == 14.2
  640. # Spool weight should be updated: 50 + 14.2 = 64.2
  641. assert spool.weight_used == 64.2