test_usage_tracker.py 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962
  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. _archive_colors_from_spools,
  12. _spool_color_to_hex,
  13. _track_from_3mf,
  14. on_print_complete,
  15. on_print_start,
  16. )
  17. def _make_spool(*, id=1, label_weight=1000, weight_used=0, tag_uid=None, tray_uuid=None, rgba=None):
  18. """Create a mock Spool object."""
  19. spool = MagicMock()
  20. spool.id = id
  21. spool.label_weight = label_weight
  22. spool.weight_used = weight_used
  23. spool.tag_uid = tag_uid
  24. spool.tray_uuid = tray_uuid
  25. spool.last_used = None
  26. spool.cost_per_kg = None
  27. spool.material = "PLA"
  28. spool.rgba = rgba
  29. return spool
  30. def _make_assignment(*, spool_id=1, printer_id=1, ams_id=0, tray_id=0, created_at=None):
  31. """Create a mock SpoolAssignment object."""
  32. assignment = MagicMock()
  33. assignment.spool_id = spool_id
  34. assignment.printer_id = printer_id
  35. assignment.ams_id = ams_id
  36. assignment.tray_id = tray_id
  37. assignment.created_at = created_at or datetime.now(timezone.utc)
  38. return assignment
  39. def _make_printer_state(ams_data, progress=0, layer_num=0, tray_now=255):
  40. """Create a mock printer state with AMS data."""
  41. state = MagicMock()
  42. state.raw_data = {"ams": ams_data}
  43. state.progress = progress
  44. state.layer_num = layer_num
  45. state.tray_now = tray_now
  46. return state
  47. def _make_printer_manager(state=None):
  48. """Create a mock printer manager."""
  49. pm = MagicMock()
  50. pm.get_status.return_value = state
  51. return pm
  52. class TestOnPrintStart:
  53. """Tests for on_print_start — capturing AMS remain%."""
  54. @pytest.fixture(autouse=True)
  55. def _clear_sessions(self):
  56. _active_sessions.clear()
  57. yield
  58. _active_sessions.clear()
  59. @pytest.mark.asyncio
  60. async def test_creates_session_with_valid_remain(self):
  61. """Session created with remain% data for trays reporting 0-100."""
  62. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 80}]}]
  63. pm = _make_printer_manager(_make_printer_state(ams_data))
  64. await on_print_start(1, {"subtask_name": "test_print"}, pm)
  65. assert 1 in _active_sessions
  66. session = _active_sessions[1]
  67. assert session.print_name == "test_print"
  68. assert session.tray_remain_start == {(0, 0): 80}
  69. @pytest.mark.asyncio
  70. async def test_creates_session_even_without_valid_remain(self):
  71. """Session still created when remain=-1 (for 3MF fallback path)."""
  72. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": -1}]}]
  73. pm = _make_printer_manager(_make_printer_state(ams_data))
  74. await on_print_start(1, {"subtask_name": "test_print"}, pm)
  75. assert 1 in _active_sessions
  76. session = _active_sessions[1]
  77. assert session.tray_remain_start == {} # Empty, no valid remain
  78. @pytest.mark.asyncio
  79. async def test_skips_without_ams_data(self):
  80. """No session created when no AMS data available."""
  81. state = MagicMock()
  82. state.raw_data = {"ams": []}
  83. pm = _make_printer_manager(state)
  84. await on_print_start(1, {"subtask_name": "test"}, pm)
  85. assert 1 not in _active_sessions
  86. class TestOnPrintCompleteAMSDelta:
  87. """Tests for Path 1: AMS remain% delta tracking."""
  88. @pytest.fixture(autouse=True)
  89. def _clear_sessions(self):
  90. _active_sessions.clear()
  91. yield
  92. _active_sessions.clear()
  93. @pytest.fixture(autouse=True)
  94. def _mock_get_setting(self):
  95. with patch(
  96. "backend.app.api.routes.settings.get_setting",
  97. new_callable=AsyncMock,
  98. return_value=None,
  99. ):
  100. yield
  101. @pytest.mark.asyncio
  102. async def test_computes_delta_and_updates_spool(self):
  103. """Spool weight_used updated by remain% delta * label_weight."""
  104. # Set up session with start remain = 80%
  105. _active_sessions[1] = PrintSession(
  106. printer_id=1,
  107. print_name="test",
  108. started_at=datetime.now(timezone.utc),
  109. tray_remain_start={(0, 0): 80},
  110. )
  111. # Current remain = 70% → 10% consumed → 100g on 1000g spool
  112. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 70}]}]
  113. pm = _make_printer_manager(_make_printer_state(ams_data))
  114. spool = _make_spool(label_weight=1000, weight_used=50)
  115. assignment = _make_assignment()
  116. db = AsyncMock()
  117. # First 2 executes → _find_3mf_by_filename (library + archive search, uses scalars().all()),
  118. # then assignment, then spool for the AMS fallback path
  119. db.execute = AsyncMock(
  120. side_effect=[
  121. MagicMock(), # _find_3mf_by_filename: library search
  122. MagicMock(), # _find_3mf_by_filename: archive search
  123. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  124. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  125. ]
  126. )
  127. results = await on_print_complete(1, {"status": "completed"}, pm, db)
  128. assert len(results) == 1
  129. assert results[0]["weight_used"] == 100.0
  130. assert results[0]["percent_used"] == 10
  131. # weight_used should be old (50) + delta (100)
  132. assert spool.weight_used == 150.0
  133. db.commit.assert_called_once()
  134. @pytest.mark.asyncio
  135. async def test_skips_negative_delta(self):
  136. """No tracking when remain increased (spool refilled)."""
  137. _active_sessions[1] = PrintSession(
  138. printer_id=1,
  139. print_name="test",
  140. started_at=datetime.now(timezone.utc),
  141. tray_remain_start={(0, 0): 50},
  142. )
  143. # Remain went UP: 50 → 80 (refilled)
  144. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 80}]}]
  145. pm = _make_printer_manager(_make_printer_state(ams_data))
  146. db = AsyncMock()
  147. results = await on_print_complete(1, {"status": "completed"}, pm, db)
  148. assert results == []
  149. db.commit.assert_not_called()
  150. @pytest.mark.asyncio
  151. async def test_no_session_falls_through_to_3mf(self):
  152. """When no session exists, AMS delta path skipped (3MF may still run)."""
  153. pm = _make_printer_manager()
  154. db = AsyncMock()
  155. results = await on_print_complete(1, {"status": "completed"}, pm, db)
  156. assert results == []
  157. @pytest.mark.asyncio
  158. async def test_skips_fallback_for_trays_outside_print_mapping(self):
  159. """#1269: swapping a spool in an UNUSED slot mid-print must NOT charge the old spool.
  160. Reproduces maugsburger's report: single-color print on AMS0-T3
  161. (ams_mapping=[3]). User swaps spools in T1 and T2 during the print —
  162. those slots report remain=0 at completion (new spool with no tag).
  163. The fallback must skip T1 and T2 because they were never in the
  164. print's tray mapping or runtime tray_change_log.
  165. """
  166. _active_sessions[1] = PrintSession(
  167. printer_id=1,
  168. print_name="splitter",
  169. started_at=datetime.now(timezone.utc),
  170. tray_remain_start={(0, 1): 100, (0, 2): 17, (0, 3): 100},
  171. tray_now_at_start=3,
  172. ams_mapping=[3],
  173. )
  174. # User swapped T1 and T2 mid-print → both report remain=0 now.
  175. # T3 was actually used but it's also at 0 now. Without the fix the
  176. # fallback would charge the originally-assigned spools at T1 and T2.
  177. ams_data = [
  178. {
  179. "id": 0,
  180. "tray": [
  181. {"id": 1, "remain": 0},
  182. {"id": 2, "remain": 0},
  183. {"id": 3, "remain": 0},
  184. ],
  185. }
  186. ]
  187. state = _make_printer_state(ams_data, tray_now=3)
  188. state.tray_change_log = [(3, 0)] # only T3 was loaded during the print
  189. pm = _make_printer_manager(state)
  190. # Only T3 should reach the spool lookup; T1 and T2 must be filtered
  191. # out before any DB query is issued for them.
  192. t3_spool = _make_spool(id=8, label_weight=1000, weight_used=0)
  193. t3_assignment = _make_assignment(spool_id=8, ams_id=0, tray_id=3)
  194. db = AsyncMock()
  195. db.execute = AsyncMock(
  196. side_effect=[
  197. MagicMock(), # _find_3mf_by_filename: library search
  198. MagicMock(), # _find_3mf_by_filename: archive search
  199. MagicMock(scalar_one_or_none=MagicMock(return_value=t3_assignment)),
  200. MagicMock(scalar_one_or_none=MagicMock(return_value=t3_spool)),
  201. ]
  202. )
  203. results = await on_print_complete(1, {"status": "completed"}, pm, db)
  204. # Only T3 should be charged. T1 (spool 27 in the report) and T2
  205. # (spool 24) must NOT appear in the results.
  206. assert len(results) == 1
  207. assert results[0]["ams_id"] == 0
  208. assert results[0]["tray_id"] == 3
  209. class TestTrackFrom3MF:
  210. """Tests for Path 2: 3MF per-filament fallback tracking."""
  211. @pytest.mark.asyncio
  212. async def test_updates_non_bl_spool_from_3mf(self):
  213. """Non-BL spool gets weight_used from 3MF used_g for completed print."""
  214. spool = _make_spool(id=5, label_weight=1000, weight_used=100)
  215. assignment = _make_assignment(spool_id=5)
  216. archive = MagicMock()
  217. archive.file_path = "archives/test.3mf"
  218. db = AsyncMock()
  219. # archive, queue_item(None), assignment, spool
  220. db.execute = AsyncMock(
  221. side_effect=[
  222. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  223. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  224. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  225. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  226. ]
  227. )
  228. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  229. filament_usage = [{"slot_id": 1, "used_g": 25.5, "type": "PLA", "color": "#FF0000"}]
  230. with (
  231. patch("backend.app.core.config.settings") as mock_settings,
  232. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  233. ):
  234. mock_path = MagicMock()
  235. mock_path.exists.return_value = True
  236. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  237. results = await _track_from_3mf(
  238. printer_id=1,
  239. archive_id=10,
  240. status="completed",
  241. print_name="test_print",
  242. handled_trays=set(),
  243. printer_manager=pm,
  244. db=db,
  245. )
  246. assert len(results) == 1
  247. assert results[0]["spool_id"] == 5
  248. assert results[0]["weight_used"] == 25.5
  249. # weight_used = old (100) + 3MF (25.5)
  250. assert spool.weight_used == 125.5
  251. @pytest.mark.asyncio
  252. async def test_scales_by_progress_for_failed_print(self):
  253. """Failed print scales 3MF estimate by progress percentage."""
  254. spool = _make_spool(id=1, label_weight=1000, weight_used=0)
  255. assignment = _make_assignment()
  256. archive = MagicMock()
  257. archive.file_path = "archives/test.3mf"
  258. db = AsyncMock()
  259. # archive, queue_item(None), assignment, spool
  260. db.execute = AsyncMock(
  261. side_effect=[
  262. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  263. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  264. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  265. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  266. ]
  267. )
  268. # Print failed at 50% progress → 50g consumed from 100g estimate
  269. pm = _make_printer_manager(_make_printer_state([], progress=50, tray_now=0))
  270. filament_usage = [{"slot_id": 1, "used_g": 100.0, "type": "PLA", "color": ""}]
  271. with (
  272. patch("backend.app.core.config.settings") as mock_settings,
  273. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  274. ):
  275. mock_path = MagicMock()
  276. mock_path.exists.return_value = True
  277. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  278. results = await _track_from_3mf(
  279. printer_id=1,
  280. archive_id=10,
  281. status="failed",
  282. print_name="test",
  283. handled_trays=set(),
  284. printer_manager=pm,
  285. db=db,
  286. )
  287. assert len(results) == 1
  288. assert results[0]["weight_used"] == 50.0
  289. assert spool.weight_used == 50.0
  290. @pytest.mark.asyncio
  291. async def test_tracks_bl_spools_via_3mf(self):
  292. """BL spools (with tag_uid) ARE now tracked via 3MF (unified tracking)."""
  293. spool = _make_spool(tag_uid="ABCD1234", tray_uuid="A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4")
  294. assignment = _make_assignment()
  295. archive = MagicMock()
  296. archive.file_path = "archives/test.3mf"
  297. db = AsyncMock()
  298. # archive, queue_item(None), assignment, spool
  299. db.execute = AsyncMock(
  300. side_effect=[
  301. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  302. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  303. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  304. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  305. ]
  306. )
  307. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  308. filament_usage = [{"slot_id": 1, "used_g": 50.0, "type": "PLA", "color": ""}]
  309. with (
  310. patch("backend.app.core.config.settings") as mock_settings,
  311. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  312. ):
  313. mock_path = MagicMock()
  314. mock_path.exists.return_value = True
  315. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  316. results = await _track_from_3mf(
  317. printer_id=1,
  318. archive_id=10,
  319. status="completed",
  320. print_name="test",
  321. handled_trays=set(),
  322. printer_manager=pm,
  323. db=db,
  324. )
  325. assert len(results) == 1
  326. assert results[0]["spool_id"] == 1
  327. assert results[0]["weight_used"] == 50.0
  328. @pytest.mark.asyncio
  329. async def test_skips_already_handled_trays(self):
  330. """Trays handled by AMS remain% delta are not double-tracked via 3MF."""
  331. archive = MagicMock()
  332. archive.file_path = "archives/test.3mf"
  333. db = AsyncMock()
  334. # archive, queue_item(None)
  335. db.execute = AsyncMock(
  336. side_effect=[
  337. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  338. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  339. ]
  340. )
  341. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  342. filament_usage = [{"slot_id": 1, "used_g": 50.0, "type": "PLA", "color": ""}]
  343. with (
  344. patch("backend.app.core.config.settings") as mock_settings,
  345. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  346. ):
  347. mock_path = MagicMock()
  348. mock_path.exists.return_value = True
  349. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  350. results = await _track_from_3mf(
  351. printer_id=1,
  352. archive_id=10,
  353. status="completed",
  354. print_name="test",
  355. handled_trays={(0, 0)}, # slot_id=1 → ams_id=0, tray_id=0
  356. printer_manager=pm,
  357. db=db,
  358. )
  359. assert results == []
  360. @pytest.mark.asyncio
  361. async def test_slot_to_tray_mapping(self):
  362. """3MF slot_id maps correctly to (ams_id, tray_id) via tray_now."""
  363. # tray_now=4 → ams_id=1, tray_id=0 (single filament uses tray_now)
  364. spool = _make_spool(id=9)
  365. assignment = _make_assignment(spool_id=9, ams_id=1, tray_id=0)
  366. archive = MagicMock()
  367. archive.file_path = "archives/test.3mf"
  368. db = AsyncMock()
  369. # archive, queue_item(None), assignment, spool
  370. db.execute = AsyncMock(
  371. side_effect=[
  372. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  373. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  374. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  375. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  376. ]
  377. )
  378. pm = _make_printer_manager(_make_printer_state([], tray_now=4))
  379. filament_usage = [{"slot_id": 5, "used_g": 30.0, "type": "PETG", "color": ""}]
  380. with (
  381. patch("backend.app.core.config.settings") as mock_settings,
  382. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  383. ):
  384. mock_path = MagicMock()
  385. mock_path.exists.return_value = True
  386. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  387. results = await _track_from_3mf(
  388. printer_id=1,
  389. archive_id=10,
  390. status="completed",
  391. print_name="test",
  392. handled_trays=set(),
  393. printer_manager=pm,
  394. db=db,
  395. )
  396. assert len(results) == 1
  397. assert results[0]["ams_id"] == 1
  398. assert results[0]["tray_id"] == 0
  399. class TestSpoolAssignmentSnapshot:
  400. """Tests for spool assignment snapshotting at print start (#459).
  401. When a spool runs empty mid-print, on_ams_change deletes the SpoolAssignment.
  402. The snapshot captured at print start ensures usage is still attributed correctly.
  403. """
  404. @pytest.fixture(autouse=True)
  405. def _clear_sessions(self):
  406. _active_sessions.clear()
  407. yield
  408. _active_sessions.clear()
  409. @pytest.fixture(autouse=True)
  410. def _mock_get_setting(self):
  411. with patch(
  412. "backend.app.api.routes.settings.get_setting",
  413. new_callable=AsyncMock,
  414. return_value=None,
  415. ):
  416. yield
  417. @pytest.mark.asyncio
  418. async def test_on_print_start_snapshots_assignments_with_db(self):
  419. """on_print_start captures spool assignments when db is provided."""
  420. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 80}, {"id": 1, "remain": 60}]}]
  421. pm = _make_printer_manager(_make_printer_state(ams_data, tray_now=0))
  422. assignment_0 = _make_assignment(spool_id=10, printer_id=1, ams_id=0, tray_id=0)
  423. assignment_1 = _make_assignment(spool_id=20, printer_id=1, ams_id=0, tray_id=1)
  424. db = AsyncMock()
  425. scalars_mock = MagicMock()
  426. scalars_mock.all.return_value = [assignment_0, assignment_1]
  427. result_mock = MagicMock()
  428. result_mock.scalars.return_value = scalars_mock
  429. db.execute = AsyncMock(return_value=result_mock)
  430. await on_print_start(1, {"subtask_name": "Benchy"}, pm, db=db)
  431. session = _active_sessions[1]
  432. assert session.spool_assignments == {(0, 0): 10, (0, 1): 20}
  433. @pytest.mark.asyncio
  434. async def test_on_print_start_empty_snapshot_without_db(self):
  435. """on_print_start creates empty snapshot when no db provided."""
  436. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 80}]}]
  437. pm = _make_printer_manager(_make_printer_state(ams_data, tray_now=0))
  438. await on_print_start(1, {"subtask_name": "Benchy"}, pm)
  439. session = _active_sessions[1]
  440. assert session.spool_assignments == {}
  441. @pytest.mark.asyncio
  442. async def test_3mf_uses_snapshot_instead_of_live_query(self):
  443. """_track_from_3mf uses snapshot spool_id without querying SpoolAssignment."""
  444. spool = _make_spool(id=42, label_weight=1000)
  445. archive = MagicMock()
  446. archive.file_path = "archives/test.3mf"
  447. # db: archive, queue_item(None), spool — NO assignment query needed
  448. db = AsyncMock()
  449. db.execute = AsyncMock(
  450. side_effect=[
  451. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  452. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  453. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  454. ]
  455. )
  456. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  457. filament_usage = [{"slot_id": 1, "used_g": 15.0, "type": "PLA", "color": "#FF0000"}]
  458. with (
  459. patch("backend.app.core.config.settings") as mock_settings,
  460. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  461. ):
  462. mock_path = MagicMock()
  463. mock_path.exists.return_value = True
  464. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  465. results = await _track_from_3mf(
  466. printer_id=1,
  467. archive_id=10,
  468. status="completed",
  469. print_name="Test",
  470. handled_trays=set(),
  471. printer_manager=pm,
  472. db=db,
  473. spool_assignments={(0, 0): 42},
  474. )
  475. assert len(results) == 1
  476. assert results[0]["spool_id"] == 42
  477. assert results[0]["weight_used"] == 15.0
  478. @pytest.mark.asyncio
  479. async def test_3mf_falls_back_to_live_query_without_snapshot(self):
  480. """_track_from_3mf queries SpoolAssignment when no snapshot exists."""
  481. spool = _make_spool(id=5, label_weight=1000)
  482. assignment = _make_assignment(spool_id=5)
  483. archive = MagicMock()
  484. archive.file_path = "archives/test.3mf"
  485. # db: archive, queue_item(None), assignment, spool
  486. db = AsyncMock()
  487. db.execute = AsyncMock(
  488. side_effect=[
  489. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  490. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  491. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  492. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  493. ]
  494. )
  495. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  496. filament_usage = [{"slot_id": 1, "used_g": 10.0, "type": "PLA", "color": "#FF0000"}]
  497. with (
  498. patch("backend.app.core.config.settings") as mock_settings,
  499. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  500. ):
  501. mock_path = MagicMock()
  502. mock_path.exists.return_value = True
  503. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  504. results = await _track_from_3mf(
  505. printer_id=1,
  506. archive_id=10,
  507. status="completed",
  508. print_name="Test",
  509. handled_trays=set(),
  510. printer_manager=pm,
  511. db=db,
  512. spool_assignments=None,
  513. )
  514. assert len(results) == 1
  515. assert results[0]["spool_id"] == 5
  516. @pytest.mark.asyncio
  517. async def test_ams_delta_uses_snapshot_over_live_query(self):
  518. """AMS remain% fallback uses snapshot spool_id instead of live query."""
  519. spool = _make_spool(id=77, label_weight=1000)
  520. _active_sessions[1] = PrintSession(
  521. printer_id=1,
  522. print_name="Benchy",
  523. started_at=datetime.now(timezone.utc),
  524. tray_remain_start={(0, 0): 80},
  525. spool_assignments={(0, 0): 77},
  526. )
  527. # Current remain = 70% → 10% delta → 100g
  528. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 70}]}]
  529. pm = _make_printer_manager(_make_printer_state(ams_data))
  530. # First 2 executes → _find_3mf_by_filename (library + archive search),
  531. # then live assignment check (returns None), then spool lookup by snapshot spool_id
  532. db = AsyncMock()
  533. db.execute = AsyncMock(
  534. side_effect=[
  535. MagicMock(), # _find_3mf_by_filename: library search
  536. MagicMock(), # _find_3mf_by_filename: archive search
  537. MagicMock(scalar_one_or_none=MagicMock(return_value=None)), # live assignment
  538. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  539. ]
  540. )
  541. results = await on_print_complete(
  542. printer_id=1,
  543. data={"status": "completed"},
  544. printer_manager=pm,
  545. db=db,
  546. archive_id=None,
  547. )
  548. assert len(results) == 1
  549. assert results[0]["spool_id"] == 77
  550. assert results[0]["weight_used"] == 100.0
  551. @pytest.mark.asyncio
  552. async def test_ams_delta_falls_back_to_live_query_without_snapshot(self):
  553. """AMS remain% fallback queries SpoolAssignment when snapshot is empty."""
  554. spool = _make_spool(id=33, label_weight=1000)
  555. assignment = _make_assignment(spool_id=33)
  556. _active_sessions[1] = PrintSession(
  557. printer_id=1,
  558. print_name="Benchy",
  559. started_at=datetime.now(timezone.utc),
  560. tray_remain_start={(0, 0): 80},
  561. spool_assignments={}, # Empty snapshot (pre-upgrade session)
  562. )
  563. ams_data = [{"id": 0, "tray": [{"id": 0, "remain": 70}]}]
  564. pm = _make_printer_manager(_make_printer_state(ams_data))
  565. # First 2 executes → _find_3mf_by_filename (library + archive search),
  566. # then assignment and spool for the AMS fallback path
  567. db = AsyncMock()
  568. db.execute = AsyncMock(
  569. side_effect=[
  570. MagicMock(), # _find_3mf_by_filename: library search
  571. MagicMock(), # _find_3mf_by_filename: archive search
  572. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  573. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  574. ]
  575. )
  576. results = await on_print_complete(
  577. printer_id=1,
  578. data={"status": "completed"},
  579. printer_manager=pm,
  580. db=db,
  581. archive_id=None,
  582. )
  583. assert len(results) == 1
  584. assert results[0]["spool_id"] == 33
  585. @pytest.mark.asyncio
  586. async def test_snapshot_survives_mid_print_unlink(self):
  587. """Core bug scenario: snapshot provides spool_id after mid-print unlink.
  588. Simulates the #459 scenario: spool runs empty mid-print, on_ams_change
  589. deletes the SpoolAssignment, but the snapshot from print start still
  590. has the spool_id so usage is correctly attributed at print completion.
  591. """
  592. spool = _make_spool(id=8, label_weight=1000, weight_used=50)
  593. archive = MagicMock()
  594. archive.file_path = "archives/big_print.3mf"
  595. # Explicit numeric so the #1344 top-up branch doesn't trip a
  596. # MagicMock-vs-float comparison.
  597. archive.filament_used_grams = 14.2
  598. # Session was created at print start WITH snapshot
  599. _active_sessions[1] = PrintSession(
  600. printer_id=1,
  601. print_name="Big Print",
  602. started_at=datetime.now(timezone.utc),
  603. tray_remain_start={(0, 0): 90},
  604. spool_assignments={(0, 0): 8}, # Snapshot from print start
  605. )
  606. pm = _make_printer_manager(
  607. _make_printer_state(
  608. [{"id": 0, "tray": [{"id": 0, "remain": 75}]}],
  609. tray_now=0,
  610. )
  611. )
  612. filament_usage = [{"slot_id": 1, "used_g": 14.2, "type": "PLA", "color": "#FF0000"}]
  613. # db: archive, queue_item(None), live assignment(None), spool,
  614. # then cost aggregation queries
  615. # NOTE: No assignment in db — it was deleted by on_ams_change mid-print!
  616. db = AsyncMock()
  617. db.execute = AsyncMock(
  618. side_effect=[
  619. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  620. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  621. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  622. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  623. # Cost-update block re-selects the archive to mutate cost.
  624. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  625. ]
  626. )
  627. with (
  628. patch("backend.app.core.config.settings") as mock_settings,
  629. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  630. ):
  631. mock_path = MagicMock()
  632. mock_path.exists.return_value = True
  633. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  634. results = await on_print_complete(
  635. printer_id=1,
  636. data={"status": "completed"},
  637. printer_manager=pm,
  638. db=db,
  639. archive_id=100,
  640. )
  641. # Usage should be tracked despite assignment being deleted mid-print
  642. assert len(results) >= 1
  643. assert results[0]["spool_id"] == 8
  644. assert results[0]["weight_used"] == 14.2
  645. # Spool weight should be updated: 50 + 14.2 = 64.2
  646. assert spool.weight_used == 64.2
  647. class TestSpoolColorToHex:
  648. """`_spool_color_to_hex` normalises Spool.rgba (RRGGBBAA, no #) to #RRGGBB."""
  649. def test_strips_alpha_and_adds_hash(self):
  650. assert _spool_color_to_hex("000000FF") == "#000000"
  651. assert _spool_color_to_hex("EC984CFF") == "#EC984C"
  652. def test_uppercases(self):
  653. assert _spool_color_to_hex("ec984cff") == "#EC984C"
  654. def test_accepts_six_char_value(self):
  655. """A value with no alpha is still valid."""
  656. assert _spool_color_to_hex("161616") == "#161616"
  657. def test_tolerates_leading_hash(self):
  658. assert _spool_color_to_hex("#000000FF") == "#000000"
  659. def test_none_and_too_short_return_none(self):
  660. """Missing / malformed colour falls back to the 3MF value."""
  661. assert _spool_color_to_hex(None) is None
  662. assert _spool_color_to_hex("") is None
  663. assert _spool_color_to_hex("FFF") is None
  664. class TestArchiveColorsFromSpools:
  665. """`_archive_colors_from_spools` rebuilds an archive's filament_color from
  666. the inventory spools that fed the print (#1494). All-or-nothing: a partial
  667. match returns None so the 3MF colour is left intact."""
  668. def test_single_slot_matched(self):
  669. """The #1494 case: one used slot, matched to a #000000 spool."""
  670. usage = [{"slot_id": 1, "used_g": 15.9, "color": "#161616"}]
  671. results = [{"slot_id": 1, "color": "#000000"}]
  672. assert _archive_colors_from_spools(usage, results) == ["#000000"]
  673. def test_multi_slot_all_matched_keeps_slot_order(self):
  674. usage = [
  675. {"slot_id": 1, "used_g": 10.0, "color": "#111111"},
  676. {"slot_id": 2, "used_g": 20.0, "color": "#222222"},
  677. ]
  678. # results deliberately out of slot order — output must be slot-ordered
  679. results = [
  680. {"slot_id": 2, "color": "#00FF00"},
  681. {"slot_id": 1, "color": "#FF0000"},
  682. ]
  683. assert _archive_colors_from_spools(usage, results) == ["#FF0000", "#00FF00"]
  684. def test_duplicate_colors_deduplicated(self):
  685. """Two slots of the same spool colour collapse to one entry, as the
  686. 3MF-derived path also de-duplicates."""
  687. usage = [
  688. {"slot_id": 1, "used_g": 10.0, "color": "#111111"},
  689. {"slot_id": 2, "used_g": 20.0, "color": "#222222"},
  690. ]
  691. results = [
  692. {"slot_id": 1, "color": "#000000"},
  693. {"slot_id": 2, "color": "#000000"},
  694. ]
  695. assert _archive_colors_from_spools(usage, results) == ["#000000"]
  696. def test_partial_match_returns_none(self):
  697. """Slot 2 was used but never matched to a spool — leave the 3MF colour
  698. untouched rather than dropping slot 2 from the archive."""
  699. usage = [
  700. {"slot_id": 1, "used_g": 10.0, "color": "#111111"},
  701. {"slot_id": 2, "used_g": 20.0, "color": "#222222"},
  702. ]
  703. results = [{"slot_id": 1, "color": "#000000"}]
  704. assert _archive_colors_from_spools(usage, results) is None
  705. def test_matched_spool_without_color_returns_none(self):
  706. """A spool with no rgba (color None) does not count as matched."""
  707. usage = [{"slot_id": 1, "used_g": 15.0, "color": "#161616"}]
  708. results = [{"slot_id": 1, "color": None}]
  709. assert _archive_colors_from_spools(usage, results) is None
  710. def test_unused_slot_not_required(self):
  711. """A slot with zero usage need not be matched."""
  712. usage = [
  713. {"slot_id": 1, "used_g": 15.0, "color": "#161616"},
  714. {"slot_id": 2, "used_g": 0.0, "color": "#888888"},
  715. ]
  716. results = [{"slot_id": 1, "color": "#000000"}]
  717. assert _archive_colors_from_spools(usage, results) == ["#000000"]
  718. def test_no_used_slots_returns_none(self):
  719. assert _archive_colors_from_spools([], []) is None
  720. def test_ams_fallback_results_excluded(self):
  721. """AMS remain%-delta fallback results carry slot_id=None and must not
  722. satisfy the match for a real 3MF slot."""
  723. usage = [{"slot_id": 1, "used_g": 15.0, "color": "#161616"}]
  724. results = [{"slot_id": None, "color": "#000000"}]
  725. assert _archive_colors_from_spools(usage, results) is None
  726. class TestArchiveFilamentColorRewrite:
  727. """`_track_from_3mf` overwrites the archive's filament_color with the
  728. matched inventory spool colour at print completion (#1494)."""
  729. @pytest.mark.asyncio
  730. async def test_archive_color_adopts_spool_color(self):
  731. """A print from a #000000 inventory spool whose 3MF says #161616 ends
  732. up with the archive showing the spool's #000000."""
  733. spool = _make_spool(id=5, label_weight=1000, weight_used=100, rgba="000000FF")
  734. assignment = _make_assignment(spool_id=5)
  735. archive = MagicMock()
  736. archive.file_path = "archives/test.3mf"
  737. archive.filament_color = "#161616" # what archive.py set from the 3MF
  738. db = AsyncMock()
  739. db.execute = AsyncMock(
  740. side_effect=[
  741. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  742. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  743. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  744. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  745. ]
  746. )
  747. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  748. filament_usage = [{"slot_id": 1, "used_g": 25.5, "type": "PETG", "color": "#161616"}]
  749. with (
  750. patch("backend.app.core.config.settings") as mock_settings,
  751. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  752. ):
  753. mock_path = MagicMock()
  754. mock_path.exists.return_value = True
  755. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  756. results = await _track_from_3mf(
  757. printer_id=1,
  758. archive_id=10,
  759. status="completed",
  760. print_name="test_print",
  761. handled_trays=set(),
  762. printer_manager=pm,
  763. db=db,
  764. )
  765. assert len(results) == 1
  766. assert results[0]["color"] == "#000000"
  767. assert results[0]["slot_id"] == 1
  768. # The archive colour was rewritten from the slicer's #161616 to the
  769. # inventory spool's #000000.
  770. assert archive.filament_color == "#000000"
  771. @pytest.mark.asyncio
  772. async def test_archive_color_untouched_when_spool_has_no_color(self):
  773. """A spool with no rgba leaves the 3MF colour in place."""
  774. spool = _make_spool(id=5, label_weight=1000, weight_used=100, rgba=None)
  775. assignment = _make_assignment(spool_id=5)
  776. archive = MagicMock()
  777. archive.file_path = "archives/test.3mf"
  778. archive.filament_color = "#161616"
  779. db = AsyncMock()
  780. db.execute = AsyncMock(
  781. side_effect=[
  782. MagicMock(scalar_one_or_none=MagicMock(return_value=archive)),
  783. MagicMock(scalar_one_or_none=MagicMock(return_value=None)),
  784. MagicMock(scalar_one_or_none=MagicMock(return_value=assignment)),
  785. MagicMock(scalar_one_or_none=MagicMock(return_value=spool)),
  786. ]
  787. )
  788. pm = _make_printer_manager(_make_printer_state([], tray_now=0))
  789. filament_usage = [{"slot_id": 1, "used_g": 25.5, "type": "PETG", "color": "#161616"}]
  790. with (
  791. patch("backend.app.core.config.settings") as mock_settings,
  792. patch("backend.app.utils.threemf_tools.extract_filament_usage_from_3mf", return_value=filament_usage),
  793. ):
  794. mock_path = MagicMock()
  795. mock_path.exists.return_value = True
  796. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  797. await _track_from_3mf(
  798. printer_id=1,
  799. archive_id=10,
  800. status="completed",
  801. print_name="test_print",
  802. handled_trays=set(),
  803. printer_manager=pm,
  804. db=db,
  805. )
  806. assert archive.filament_color == "#161616"