test_usage_tracker.py 47 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260
  1. """Unit tests for usage_tracker.py — 3MF-primary filament tracking.
  2. Tests the unified tracking logic: 3MF slicer estimates as primary path,
  3. AMS remain% delta as fallback, per-layer gcode for partial prints,
  4. slot-to-tray mapping resolution, and notification variable formatting.
  5. """
  6. from datetime import datetime, timezone
  7. from types import SimpleNamespace
  8. from unittest.mock import AsyncMock, MagicMock, patch
  9. import pytest
  10. from backend.app.services.usage_tracker import (
  11. PrintSession,
  12. _active_sessions,
  13. _decode_mqtt_mapping,
  14. _match_slots_by_color,
  15. _track_from_3mf,
  16. on_print_complete,
  17. on_print_start,
  18. )
  19. def _make_spool(spool_id=1, label_weight=1000, weight_used=0, tag_uid=None, tray_uuid=None):
  20. """Create a mock Spool object."""
  21. spool = MagicMock()
  22. spool.id = spool_id
  23. spool.label_weight = label_weight
  24. spool.weight_used = weight_used
  25. spool.tag_uid = tag_uid
  26. spool.tray_uuid = tray_uuid
  27. spool.last_used = None
  28. spool.cost_per_kg = None
  29. spool.material = "PLA"
  30. return spool
  31. def _make_assignment(spool_id=1, printer_id=1, ams_id=0, tray_id=0):
  32. """Create a mock SpoolAssignment object."""
  33. assignment = MagicMock()
  34. assignment.spool_id = spool_id
  35. assignment.printer_id = printer_id
  36. assignment.ams_id = ams_id
  37. assignment.tray_id = tray_id
  38. return assignment
  39. def _make_archive(archive_id=1, file_path="archives/1/test.3mf", extra_data=None):
  40. """Create a mock PrintArchive object."""
  41. archive = MagicMock()
  42. archive.id = archive_id
  43. archive.file_path = file_path
  44. archive.extra_data = extra_data
  45. return archive
  46. def _make_queue_item(ams_mapping=None, status="printing"):
  47. """Create a mock PrintQueueItem object."""
  48. item = MagicMock()
  49. item.ams_mapping = ams_mapping
  50. item.status = status
  51. return item
  52. def _mock_db_execute(*return_values):
  53. """Create a mock db with execute() that returns values in sequence."""
  54. db = AsyncMock()
  55. results = []
  56. for val in return_values:
  57. result = MagicMock()
  58. result.scalar_one_or_none.return_value = val
  59. results.append(result)
  60. db.execute = AsyncMock(side_effect=results)
  61. return db
  62. def _mock_db_sequential(responses):
  63. """Create mock db that returns responses in order."""
  64. db = AsyncMock()
  65. call_count = [0]
  66. async def mock_execute(*args, **kwargs):
  67. idx = call_count[0]
  68. call_count[0] += 1
  69. result = MagicMock()
  70. if idx < len(responses):
  71. result.scalar_one_or_none.return_value = responses[idx]
  72. else:
  73. result.scalar_one_or_none.return_value = None
  74. # For cost aggregation queries that use .scalar() instead of .scalar_one_or_none()
  75. result.scalar.return_value = None
  76. return result
  77. db.execute = mock_execute
  78. return db
  79. class TestOnPrintStart:
  80. """Tests for on_print_start()."""
  81. @pytest.fixture(autouse=True)
  82. def _clear_sessions(self):
  83. _active_sessions.clear()
  84. yield
  85. _active_sessions.clear()
  86. @pytest.mark.asyncio
  87. async def test_captures_remain_data(self):
  88. """Captures AMS remain% at print start."""
  89. printer_manager = MagicMock()
  90. printer_manager.get_status.return_value = SimpleNamespace(
  91. raw_data={"ams": [{"id": 0, "tray": [{"id": 0, "remain": 80}, {"id": 1, "remain": 50}]}]},
  92. tray_now=5,
  93. )
  94. await on_print_start(1, {"subtask_name": "Benchy"}, printer_manager)
  95. assert 1 in _active_sessions
  96. session = _active_sessions[1]
  97. assert session.print_name == "Benchy"
  98. assert session.tray_remain_start == {(0, 0): 80, (0, 1): 50}
  99. @pytest.mark.asyncio
  100. async def test_captures_tray_now_at_start(self):
  101. """Captures tray_now at print start for later use in usage tracking."""
  102. printer_manager = MagicMock()
  103. printer_manager.get_status.return_value = SimpleNamespace(
  104. raw_data={"ams": [{"id": 0, "tray": [{"id": 0, "remain": 80}]}]},
  105. tray_now=9,
  106. )
  107. await on_print_start(1, {"subtask_name": "Test"}, printer_manager)
  108. assert _active_sessions[1].tray_now_at_start == 9
  109. @pytest.mark.asyncio
  110. async def test_tray_now_at_start_255_when_unloaded(self):
  111. """Captures tray_now=255 when printer has no filament loaded at start."""
  112. printer_manager = MagicMock()
  113. printer_manager.get_status.return_value = SimpleNamespace(
  114. raw_data={"ams": [{"id": 0, "tray": [{"id": 0, "remain": 80}]}]},
  115. tray_now=255,
  116. )
  117. await on_print_start(1, {"subtask_name": "Test"}, printer_manager)
  118. assert _active_sessions[1].tray_now_at_start == 255
  119. @pytest.mark.asyncio
  120. async def test_creates_session_without_remain(self):
  121. """Creates session even without valid remain data (for 3MF tracking)."""
  122. printer_manager = MagicMock()
  123. printer_manager.get_status.return_value = SimpleNamespace(
  124. raw_data={"ams": [{"id": 0, "tray": [{"id": 0, "remain": -1}]}]},
  125. tray_now=255,
  126. )
  127. await on_print_start(1, {"subtask_name": "Test"}, printer_manager)
  128. assert 1 in _active_sessions
  129. assert _active_sessions[1].tray_remain_start == {}
  130. class TestOnPrintComplete:
  131. """Tests for on_print_complete() — path ordering and interaction."""
  132. @pytest.fixture(autouse=True)
  133. def _clear_sessions(self):
  134. _active_sessions.clear()
  135. yield
  136. _active_sessions.clear()
  137. @pytest.fixture(autouse=True)
  138. def _mock_get_setting(self):
  139. with patch(
  140. "backend.app.api.routes.settings.get_setting",
  141. new_callable=AsyncMock,
  142. return_value=None,
  143. ):
  144. yield
  145. @pytest.mark.asyncio
  146. async def test_bl_spool_uses_3mf(self):
  147. """BL spool (with tag_uid) is tracked via 3MF, not just AMS delta."""
  148. spool = _make_spool(spool_id=1, tag_uid="AABB1122", label_weight=1000)
  149. assignment = _make_assignment(spool_id=1, printer_id=1, ams_id=0, tray_id=0)
  150. archive = _make_archive(archive_id=10)
  151. # Setup: session with AMS remain data
  152. _active_sessions[1] = PrintSession(
  153. printer_id=1,
  154. print_name="Benchy",
  155. started_at=datetime.now(timezone.utc),
  156. tray_remain_start={(0, 0): 80},
  157. )
  158. # Mock printer state: tray_now=0 (AMS0-T0), single filament
  159. printer_manager = MagicMock()
  160. printer_manager.get_status.return_value = SimpleNamespace(
  161. raw_data={"ams": [{"id": 0, "tray": [{"id": 0, "remain": 70}]}]},
  162. progress=100,
  163. layer_num=50,
  164. tray_now=0,
  165. )
  166. # db returns: archive, queue_item(None), assignment, spool
  167. db = _mock_db_sequential([archive, None, assignment, spool])
  168. filament_usage = [{"slot_id": 1, "used_g": 15.0, "type": "PLA", "color": "#FF0000"}]
  169. with (
  170. patch("backend.app.core.config.settings") as mock_settings,
  171. patch(
  172. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  173. return_value=filament_usage,
  174. ),
  175. ):
  176. mock_settings.base_dir = MagicMock()
  177. mock_path = MagicMock()
  178. mock_path.exists.return_value = True
  179. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  180. results = await on_print_complete(
  181. printer_id=1,
  182. data={"status": "completed"},
  183. printer_manager=printer_manager,
  184. db=db,
  185. archive_id=10,
  186. )
  187. # 3MF path should handle it (BL guard removed)
  188. assert len(results) >= 1
  189. assert results[0]["spool_id"] == 1
  190. assert results[0]["weight_used"] == 15.0
  191. @pytest.mark.asyncio
  192. async def test_ams_delta_fallback_no_archive(self):
  193. """AMS delta tracks consumption when archive_id is None."""
  194. spool = _make_spool(spool_id=2, label_weight=1000)
  195. assignment = _make_assignment(spool_id=2)
  196. _active_sessions[1] = PrintSession(
  197. printer_id=1,
  198. print_name="Test",
  199. started_at=datetime.now(timezone.utc),
  200. tray_remain_start={(0, 0): 80},
  201. )
  202. printer_manager = MagicMock()
  203. printer_manager.get_status.return_value = SimpleNamespace(
  204. raw_data={"ams": [{"id": 0, "tray": [{"id": 0, "remain": 70}]}]},
  205. tray_now=0,
  206. last_loaded_tray=-1,
  207. )
  208. # db returns assignment then spool
  209. db = _mock_db_sequential([assignment, spool])
  210. results = await on_print_complete(
  211. printer_id=1,
  212. data={"status": "completed"},
  213. printer_manager=printer_manager,
  214. db=db,
  215. archive_id=None,
  216. )
  217. assert len(results) == 1
  218. assert results[0]["spool_id"] == 2
  219. # 10% of 1000g = 100g
  220. assert results[0]["weight_used"] == 100.0
  221. assert results[0]["percent_used"] == 10
  222. @pytest.mark.asyncio
  223. async def test_no_double_tracking(self):
  224. """When 3MF handles a tray, AMS delta skips it."""
  225. spool = _make_spool(spool_id=1, label_weight=1000)
  226. assignment = _make_assignment(spool_id=1)
  227. archive = _make_archive(archive_id=10)
  228. _active_sessions[1] = PrintSession(
  229. printer_id=1,
  230. print_name="Benchy",
  231. started_at=datetime.now(timezone.utc),
  232. tray_remain_start={(0, 0): 80},
  233. )
  234. # tray_now=0 matches the single filament slot
  235. printer_manager = MagicMock()
  236. printer_manager.get_status.return_value = SimpleNamespace(
  237. raw_data={"ams": [{"id": 0, "tray": [{"id": 0, "remain": 70}]}]},
  238. progress=100,
  239. layer_num=50,
  240. tray_now=0,
  241. )
  242. # db returns: archive, queue_item(None), assignment, spool
  243. db = _mock_db_sequential([archive, None, assignment, spool])
  244. filament_usage = [{"slot_id": 1, "used_g": 15.0, "type": "PLA", "color": "#FF0000"}]
  245. with (
  246. patch("backend.app.core.config.settings") as mock_settings,
  247. patch(
  248. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  249. return_value=filament_usage,
  250. ),
  251. ):
  252. mock_settings.base_dir = MagicMock()
  253. mock_path = MagicMock()
  254. mock_path.exists.return_value = True
  255. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  256. results = await on_print_complete(
  257. printer_id=1,
  258. data={"status": "completed"},
  259. printer_manager=printer_manager,
  260. db=db,
  261. archive_id=10,
  262. )
  263. # Only 1 result (3MF), NOT 2 (3MF + AMS delta)
  264. assert len(results) == 1
  265. assert results[0]["weight_used"] == 15.0
  266. class TestTrackFrom3mf:
  267. """Tests for _track_from_3mf() — per-layer, linear scaling, and slot mapping."""
  268. @pytest.mark.asyncio
  269. async def test_linear_fallback_for_partial_print(self):
  270. """Falls back to linear scaling when gcode layer data unavailable."""
  271. spool = _make_spool(spool_id=1, label_weight=1000)
  272. assignment = _make_assignment(spool_id=1)
  273. archive = _make_archive(archive_id=10)
  274. # db: archive, queue_item(None), assignment, spool
  275. db = _mock_db_sequential([archive, None, assignment, spool])
  276. printer_manager = MagicMock()
  277. printer_manager.get_status.return_value = SimpleNamespace(
  278. progress=50,
  279. layer_num=25,
  280. tray_now=0,
  281. )
  282. filament_usage = [{"slot_id": 1, "used_g": 20.0, "type": "PLA", "color": ""}]
  283. handled_trays: set[tuple[int, int]] = set()
  284. with (
  285. patch("backend.app.core.config.settings") as mock_settings,
  286. patch(
  287. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  288. return_value=filament_usage,
  289. ),
  290. patch(
  291. "backend.app.utils.threemf_tools.extract_layer_filament_usage_from_3mf",
  292. return_value=None, # No layer data available
  293. ),
  294. ):
  295. mock_settings.base_dir = MagicMock()
  296. mock_path = MagicMock()
  297. mock_path.exists.return_value = True
  298. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  299. results = await _track_from_3mf(
  300. printer_id=1,
  301. archive_id=10,
  302. status="failed",
  303. print_name="Benchy",
  304. handled_trays=handled_trays,
  305. printer_manager=printer_manager,
  306. db=db,
  307. )
  308. assert len(results) == 1
  309. # 50% of 20g = 10g
  310. assert results[0]["weight_used"] == 10.0
  311. # Tray should be marked as handled
  312. assert (0, 0) in handled_trays
  313. @pytest.mark.asyncio
  314. async def test_per_layer_partial_print(self):
  315. """Failed print at layer N uses gcode cumulative data."""
  316. spool = _make_spool(spool_id=1, label_weight=1000)
  317. assignment = _make_assignment(spool_id=1)
  318. archive = _make_archive(archive_id=10)
  319. # db: archive, queue_item(None), assignment, spool
  320. db = _mock_db_sequential([archive, None, assignment, spool])
  321. printer_manager = MagicMock()
  322. printer_manager.get_status.return_value = SimpleNamespace(
  323. progress=50,
  324. layer_num=25,
  325. tray_now=0,
  326. )
  327. filament_usage = [{"slot_id": 1, "used_g": 20.0, "type": "PLA", "color": ""}]
  328. # Per-layer data: at layer 25, filament 0 used 5000mm
  329. layer_data = {10: {0: 2000.0}, 25: {0: 5000.0}, 50: {0: 10000.0}}
  330. filament_props = {1: {"density": 1.24, "diameter": 1.75}}
  331. handled_trays: set[tuple[int, int]] = set()
  332. with (
  333. patch("backend.app.core.config.settings") as mock_settings,
  334. patch(
  335. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  336. return_value=filament_usage,
  337. ),
  338. patch(
  339. "backend.app.utils.threemf_tools.extract_layer_filament_usage_from_3mf",
  340. return_value=layer_data,
  341. ),
  342. patch(
  343. "backend.app.utils.threemf_tools.get_cumulative_usage_at_layer",
  344. return_value={0: 5000.0},
  345. ),
  346. patch(
  347. "backend.app.utils.threemf_tools.extract_filament_properties_from_3mf",
  348. return_value=filament_props,
  349. ),
  350. patch(
  351. "backend.app.utils.threemf_tools.mm_to_grams",
  352. return_value=12.0, # 5000mm at 1.75mm/1.24g/cm3
  353. ),
  354. ):
  355. mock_settings.base_dir = MagicMock()
  356. mock_path = MagicMock()
  357. mock_path.exists.return_value = True
  358. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  359. results = await _track_from_3mf(
  360. printer_id=1,
  361. archive_id=10,
  362. status="failed",
  363. print_name="Benchy",
  364. handled_trays=handled_trays,
  365. printer_manager=printer_manager,
  366. db=db,
  367. )
  368. assert len(results) == 1
  369. # Should use per-layer grams (12.0g), not linear scale (10.0g)
  370. assert results[0]["weight_used"] == 12.0
  371. @pytest.mark.asyncio
  372. async def test_completed_print_uses_full_weight(self):
  373. """Completed print uses full 3MF weight (scale=1.0)."""
  374. spool = _make_spool(spool_id=1, label_weight=1000)
  375. assignment = _make_assignment(spool_id=1)
  376. archive = _make_archive(archive_id=10)
  377. # db: archive, queue_item(None), assignment, spool
  378. db = _mock_db_sequential([archive, None, assignment, spool])
  379. printer_manager = MagicMock()
  380. printer_manager.get_status.return_value = SimpleNamespace(
  381. progress=100,
  382. layer_num=50,
  383. tray_now=0,
  384. )
  385. filament_usage = [{"slot_id": 1, "used_g": 20.0, "type": "PLA", "color": ""}]
  386. handled_trays: set[tuple[int, int]] = set()
  387. with (
  388. patch("backend.app.core.config.settings") as mock_settings,
  389. patch(
  390. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  391. return_value=filament_usage,
  392. ),
  393. ):
  394. mock_settings.base_dir = MagicMock()
  395. mock_path = MagicMock()
  396. mock_path.exists.return_value = True
  397. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  398. results = await _track_from_3mf(
  399. printer_id=1,
  400. archive_id=10,
  401. status="completed",
  402. print_name="Benchy",
  403. handled_trays=handled_trays,
  404. printer_manager=printer_manager,
  405. db=db,
  406. )
  407. assert len(results) == 1
  408. assert results[0]["weight_used"] == 20.0
  409. @pytest.mark.asyncio
  410. async def test_tray_now_override_for_single_filament(self):
  411. """Single-filament non-queue print uses tray_now instead of slot_id mapping."""
  412. # Spool 2 is at AMS1-T3 (global_tray_id=7)
  413. spool = _make_spool(spool_id=2, label_weight=1000)
  414. assignment = _make_assignment(spool_id=2, ams_id=1, tray_id=3)
  415. archive = _make_archive(archive_id=10)
  416. # db: archive, queue_item(None), assignment, spool
  417. db = _mock_db_sequential([archive, None, assignment, spool])
  418. # tray_now=7 = (ams_id=1, tray_id=3), the ACTUAL tray used
  419. printer_manager = MagicMock()
  420. printer_manager.get_status.return_value = SimpleNamespace(
  421. progress=100,
  422. layer_num=50,
  423. tray_now=7,
  424. )
  425. # 3MF has slot_id=12 (would default-map to ams_id=2, tray_id=3 — WRONG)
  426. filament_usage = [{"slot_id": 12, "used_g": 10.6, "type": "PLA", "color": "#FF0000"}]
  427. handled_trays: set[tuple[int, int]] = set()
  428. with (
  429. patch("backend.app.core.config.settings") as mock_settings,
  430. patch(
  431. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  432. return_value=filament_usage,
  433. ),
  434. ):
  435. mock_settings.base_dir = MagicMock()
  436. mock_path = MagicMock()
  437. mock_path.exists.return_value = True
  438. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  439. results = await _track_from_3mf(
  440. printer_id=1,
  441. archive_id=10,
  442. status="completed",
  443. print_name="Test",
  444. handled_trays=handled_trays,
  445. printer_manager=printer_manager,
  446. db=db,
  447. )
  448. assert len(results) == 1
  449. assert results[0]["spool_id"] == 2
  450. assert results[0]["ams_id"] == 1
  451. assert results[0]["tray_id"] == 3
  452. assert results[0]["weight_used"] == 10.6
  453. assert (1, 3) in handled_trays
  454. @pytest.mark.asyncio
  455. async def test_queue_ams_mapping_overrides_default(self):
  456. """Queue item ams_mapping overrides default slot_id mapping."""
  457. # Spool at AMS1-T3 (global_tray_id=7)
  458. spool = _make_spool(spool_id=5, label_weight=1000)
  459. assignment = _make_assignment(spool_id=5, ams_id=1, tray_id=3)
  460. archive = _make_archive(archive_id=20)
  461. # Queue item maps slot 1 → global tray 7 (ams_id=1, tray_id=3)
  462. queue_item = _make_queue_item(ams_mapping="[7, -1, -1, -1]")
  463. # db: archive, queue_item, assignment, spool
  464. db = _mock_db_sequential([archive, queue_item, assignment, spool])
  465. printer_manager = MagicMock()
  466. printer_manager.get_status.return_value = SimpleNamespace(
  467. progress=100,
  468. layer_num=50,
  469. tray_now=7,
  470. )
  471. filament_usage = [{"slot_id": 1, "used_g": 25.0, "type": "PETG", "color": ""}]
  472. handled_trays: set[tuple[int, int]] = set()
  473. with (
  474. patch("backend.app.core.config.settings") as mock_settings,
  475. patch(
  476. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  477. return_value=filament_usage,
  478. ),
  479. ):
  480. mock_settings.base_dir = MagicMock()
  481. mock_path = MagicMock()
  482. mock_path.exists.return_value = True
  483. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  484. results = await _track_from_3mf(
  485. printer_id=1,
  486. archive_id=20,
  487. status="completed",
  488. print_name="Queue Print",
  489. handled_trays=handled_trays,
  490. printer_manager=printer_manager,
  491. db=db,
  492. )
  493. assert len(results) == 1
  494. assert results[0]["spool_id"] == 5
  495. assert results[0]["ams_id"] == 1
  496. assert results[0]["tray_id"] == 3
  497. assert results[0]["weight_used"] == 25.0
  498. @pytest.mark.asyncio
  499. async def test_multi_filament_uses_queue_mapping(self):
  500. """Multi-filament queue prints use ams_mapping for each slot."""
  501. spool_a = _make_spool(spool_id=1, label_weight=1000)
  502. spool_b = _make_spool(spool_id=2, label_weight=1000)
  503. assign_a = _make_assignment(spool_id=1, ams_id=0, tray_id=0)
  504. assign_b = _make_assignment(spool_id=2, ams_id=1, tray_id=2)
  505. archive = _make_archive(archive_id=30)
  506. # slot 1 → tray 0 (AMS0-T0), slot 2 → tray 6 (AMS1-T2)
  507. queue_item = _make_queue_item(ams_mapping="[0, 6]")
  508. # db: archive, queue_item, assign_a, spool_a, assign_b, spool_b
  509. db = _mock_db_sequential([archive, queue_item, assign_a, spool_a, assign_b, spool_b])
  510. printer_manager = MagicMock()
  511. printer_manager.get_status.return_value = SimpleNamespace(
  512. progress=100,
  513. layer_num=50,
  514. tray_now=6,
  515. )
  516. filament_usage = [
  517. {"slot_id": 1, "used_g": 10.0, "type": "PLA", "color": ""},
  518. {"slot_id": 2, "used_g": 5.0, "type": "PETG", "color": ""},
  519. ]
  520. handled_trays: set[tuple[int, int]] = set()
  521. with (
  522. patch("backend.app.core.config.settings") as mock_settings,
  523. patch(
  524. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  525. return_value=filament_usage,
  526. ),
  527. ):
  528. mock_settings.base_dir = MagicMock()
  529. mock_path = MagicMock()
  530. mock_path.exists.return_value = True
  531. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  532. results = await _track_from_3mf(
  533. printer_id=1,
  534. archive_id=30,
  535. status="completed",
  536. print_name="Multi",
  537. handled_trays=handled_trays,
  538. printer_manager=printer_manager,
  539. db=db,
  540. )
  541. assert len(results) == 2
  542. assert results[0]["spool_id"] == 1
  543. assert results[0]["ams_id"] == 0
  544. assert results[0]["tray_id"] == 0
  545. assert results[0]["weight_used"] == 10.0
  546. assert results[1]["spool_id"] == 2
  547. assert results[1]["ams_id"] == 1
  548. assert results[1]["tray_id"] == 2
  549. assert results[1]["weight_used"] == 5.0
  550. @pytest.mark.asyncio
  551. async def test_no_tray_now_override_for_multi_filament(self):
  552. """Multi-filament non-queue prints fall back to default mapping, not tray_now."""
  553. spool = _make_spool(spool_id=1, label_weight=1000)
  554. assignment = _make_assignment(spool_id=1, ams_id=0, tray_id=0)
  555. archive = _make_archive(archive_id=10)
  556. # db: archive, queue_item(None), assignment, spool (2nd slot has no assignment)
  557. db = _mock_db_sequential([archive, None, assignment, spool, None])
  558. printer_manager = MagicMock()
  559. printer_manager.get_status.return_value = SimpleNamespace(
  560. progress=100,
  561. layer_num=50,
  562. tray_now=4, # tray_now won't be used
  563. )
  564. # Two filament slots with usage
  565. filament_usage = [
  566. {"slot_id": 1, "used_g": 10.0, "type": "PLA", "color": ""},
  567. {"slot_id": 2, "used_g": 5.0, "type": "PETG", "color": ""},
  568. ]
  569. handled_trays: set[tuple[int, int]] = set()
  570. with (
  571. patch("backend.app.core.config.settings") as mock_settings,
  572. patch(
  573. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  574. return_value=filament_usage,
  575. ),
  576. ):
  577. mock_settings.base_dir = MagicMock()
  578. mock_path = MagicMock()
  579. mock_path.exists.return_value = True
  580. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  581. results = await _track_from_3mf(
  582. printer_id=1,
  583. archive_id=10,
  584. status="completed",
  585. print_name="Test",
  586. handled_trays=handled_trays,
  587. printer_manager=printer_manager,
  588. db=db,
  589. )
  590. # Should use default mapping (slot 1 → tray 0, slot 2 → tray 1)
  591. assert len(results) == 1 # Only slot 1 has assignment
  592. assert results[0]["ams_id"] == 0
  593. assert results[0]["tray_id"] == 0
  594. @pytest.mark.asyncio
  595. async def test_stored_ams_mapping_overrides_all(self):
  596. """Stored ams_mapping from print command takes priority over queue and tray_now."""
  597. # Spool at AMS2-T1 (global_tray_id=9)
  598. spool = _make_spool(spool_id=10, label_weight=1000)
  599. assignment = _make_assignment(spool_id=10, ams_id=2, tray_id=1)
  600. archive = _make_archive(archive_id=50)
  601. # db: archive, assignment, spool (no queue lookup when ams_mapping provided)
  602. db = _mock_db_sequential([archive, assignment, spool])
  603. printer_manager = MagicMock()
  604. printer_manager.get_status.return_value = SimpleNamespace(
  605. progress=100,
  606. layer_num=50,
  607. tray_now=0, # Different from mapped tray — should be ignored
  608. last_loaded_tray=0,
  609. )
  610. filament_usage = [{"slot_id": 2, "used_g": 1.57, "type": "PLA", "color": "#FFFFFF"}]
  611. handled_trays: set[tuple[int, int]] = set()
  612. with (
  613. patch("backend.app.core.config.settings") as mock_settings,
  614. patch(
  615. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  616. return_value=filament_usage,
  617. ),
  618. ):
  619. mock_settings.base_dir = MagicMock()
  620. mock_path = MagicMock()
  621. mock_path.exists.return_value = True
  622. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  623. # ams_mapping: slot 2 (index 1) -> tray 9 (AMS2-T1)
  624. results = await _track_from_3mf(
  625. printer_id=1,
  626. archive_id=50,
  627. status="completed",
  628. print_name="Test",
  629. handled_trays=handled_trays,
  630. printer_manager=printer_manager,
  631. db=db,
  632. ams_mapping=[-1, 9],
  633. )
  634. assert len(results) == 1
  635. assert results[0]["spool_id"] == 10
  636. assert results[0]["ams_id"] == 2
  637. assert results[0]["tray_id"] == 1
  638. assert results[0]["weight_used"] == 1.6 # rounded
  639. @pytest.mark.asyncio
  640. async def test_last_loaded_tray_fallback(self):
  641. """Falls back to last_loaded_tray when tray_now_at_start and current tray_now are both 255."""
  642. # Spool at AMS2-T1 (global_tray_id=9)
  643. spool = _make_spool(spool_id=11, label_weight=1000)
  644. assignment = _make_assignment(spool_id=11, ams_id=2, tray_id=1)
  645. archive = _make_archive(archive_id=60)
  646. # db: archive, queue_item(None), assignment, spool
  647. db = _mock_db_sequential([archive, None, assignment, spool])
  648. # H2D scenario: tray_now=255 at completion, but last_loaded_tray=9
  649. printer_manager = MagicMock()
  650. printer_manager.get_status.return_value = SimpleNamespace(
  651. progress=100,
  652. layer_num=50,
  653. tray_now=255,
  654. last_loaded_tray=9,
  655. )
  656. filament_usage = [{"slot_id": 6, "used_g": 1.52, "type": "PLA", "color": "#7CC4D5"}]
  657. handled_trays: set[tuple[int, int]] = set()
  658. with (
  659. patch("backend.app.core.config.settings") as mock_settings,
  660. patch(
  661. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  662. return_value=filament_usage,
  663. ),
  664. ):
  665. mock_settings.base_dir = MagicMock()
  666. mock_path = MagicMock()
  667. mock_path.exists.return_value = True
  668. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  669. results = await _track_from_3mf(
  670. printer_id=1,
  671. archive_id=60,
  672. status="completed",
  673. print_name="Cube",
  674. handled_trays=handled_trays,
  675. printer_manager=printer_manager,
  676. db=db,
  677. tray_now_at_start=255, # H2D: 255 at start too
  678. )
  679. assert len(results) == 1
  680. assert results[0]["spool_id"] == 11
  681. assert results[0]["ams_id"] == 2
  682. assert results[0]["tray_id"] == 1
  683. @pytest.mark.asyncio
  684. async def test_tray_now_at_start_preferred_over_last_loaded(self):
  685. """tray_now_at_start is used before last_loaded_tray fallback."""
  686. spool = _make_spool(spool_id=3, label_weight=1000)
  687. assignment = _make_assignment(spool_id=3, ams_id=1, tray_id=1)
  688. archive = _make_archive(archive_id=70)
  689. db = _mock_db_sequential([archive, None, assignment, spool])
  690. # tray_now_at_start=5 (valid), last_loaded_tray=9 (different) — should use 5
  691. printer_manager = MagicMock()
  692. printer_manager.get_status.return_value = SimpleNamespace(
  693. progress=100,
  694. layer_num=50,
  695. tray_now=255,
  696. last_loaded_tray=9,
  697. )
  698. filament_usage = [{"slot_id": 1, "used_g": 5.0, "type": "PLA", "color": ""}]
  699. handled_trays: set[tuple[int, int]] = set()
  700. with (
  701. patch("backend.app.core.config.settings") as mock_settings,
  702. patch(
  703. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  704. return_value=filament_usage,
  705. ),
  706. ):
  707. mock_settings.base_dir = MagicMock()
  708. mock_path = MagicMock()
  709. mock_path.exists.return_value = True
  710. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  711. results = await _track_from_3mf(
  712. printer_id=1,
  713. archive_id=70,
  714. status="completed",
  715. print_name="Test",
  716. handled_trays=handled_trays,
  717. printer_manager=printer_manager,
  718. db=db,
  719. tray_now_at_start=5, # AMS1-T1
  720. )
  721. assert len(results) == 1
  722. assert results[0]["ams_id"] == 1
  723. assert results[0]["tray_id"] == 1
  724. class TestDecodeMqttMapping:
  725. """Tests for _decode_mqtt_mapping() — snow-encoded MQTT mapping to global tray IDs."""
  726. def test_none_input(self):
  727. assert _decode_mqtt_mapping(None) is None
  728. def test_empty_list(self):
  729. assert _decode_mqtt_mapping([]) is None
  730. def test_all_unmapped(self):
  731. """All 65535 values → None (no valid mappings)."""
  732. assert _decode_mqtt_mapping([65535, 65535, 65535]) is None
  733. def test_single_ams_slots(self):
  734. """AMS 0 slots: snow values 0-3 → global tray IDs 0-3."""
  735. assert _decode_mqtt_mapping([0, 1, 2, 3]) == [0, 1, 2, 3]
  736. def test_multi_ams_slots(self):
  737. """AMS 1 (hw_id=1): snow 256=AMS1-T0, 257=AMS1-T1 → global 4, 5."""
  738. assert _decode_mqtt_mapping([256, 257]) == [4, 5]
  739. def test_ams_ht_slot(self):
  740. """AMS-HT (hw_id=128): snow 32768 → global 128."""
  741. assert _decode_mqtt_mapping([32768]) == [128]
  742. def test_external_spool(self):
  743. """External spool: ams_hw_id=254, slot=0 → global 254."""
  744. # snow = 254 * 256 + 0 = 65024
  745. assert _decode_mqtt_mapping([65024]) == [254]
  746. def test_mixed_with_unmapped(self):
  747. """Mix of valid and unmapped (65535) values."""
  748. result = _decode_mqtt_mapping([1, 65535, 0])
  749. assert result == [1, -1, 0]
  750. def test_h2c_real_mapping(self):
  751. """Real H2C mapping from MQTT logs: [1, 0, 65535*4, 32768]."""
  752. mapping = [1, 0, 65535, 65535, 65535, 65535, 32768]
  753. result = _decode_mqtt_mapping(mapping)
  754. assert result == [1, 0, -1, -1, -1, -1, 128]
  755. def test_non_int_values_treated_as_unmapped(self):
  756. """Non-integer values in the mapping are treated as unmapped."""
  757. assert _decode_mqtt_mapping(["foo", 0]) == [-1, 0]
  758. class TestMatchSlotsByColor:
  759. """Tests for _match_slots_by_color() — color-based filament slot to AMS tray matching."""
  760. def _ams(self, trays):
  761. """Build AMS data from list of (ams_id, tray_id, color_hex, tray_type) tuples."""
  762. units: dict[int, list] = {}
  763. for ams_id, tray_id, color, tray_type in trays:
  764. units.setdefault(ams_id, []).append({"id": tray_id, "tray_color": color, "tray_type": tray_type})
  765. return [{"id": aid, "tray": t} for aid, t in units.items()]
  766. def _usage(self, slots):
  767. """Build filament_usage from list of (slot_id, color_hex) tuples."""
  768. return [{"slot_id": sid, "used_g": 10.0, "type": "PLA", "color": color} for sid, color in slots]
  769. def test_none_inputs(self):
  770. assert _match_slots_by_color(None, None) is None
  771. assert _match_slots_by_color([], None) is None
  772. assert _match_slots_by_color(None, {"ams": []}) is None
  773. def test_empty_ams(self):
  774. usage = self._usage([(1, "#FF0000")])
  775. assert _match_slots_by_color(usage, {"ams": []}) is None
  776. def test_single_slot_single_tray(self):
  777. """One 3MF slot matches one AMS tray by color."""
  778. ams = self._ams([(0, 0, "FF0000FF", "PLA")])
  779. usage = self._usage([(1, "#FF0000")])
  780. assert _match_slots_by_color(usage, {"ams": ams}) == [0]
  781. def test_a1_mini_three_colors(self):
  782. """A1 Mini: 3 slots match 3 distinct AMS trays."""
  783. ams = self._ams(
  784. [
  785. (0, 0, "FF0000FF", "PLA"), # Red
  786. (0, 1, "00FF00FF", "PLA"), # Green
  787. (0, 2, "0000FFFF", "PLA"), # Blue
  788. ]
  789. )
  790. usage = self._usage([(1, "#FF0000"), (2, "#00FF00"), (3, "#0000FF")])
  791. assert _match_slots_by_color(usage, {"ams": ams}) == [0, 1, 2]
  792. def test_dual_ams_p2s_like(self):
  793. """P2S with dual AMS: slots from second AMS unit."""
  794. ams = self._ams(
  795. [
  796. (0, 0, "AAAAAAFF", "PLA"),
  797. (0, 1, "BBBBBBFF", "PLA"),
  798. (1, 0, "CC0000FF", "PETG"), # global_id=4
  799. (1, 1, "00CC00FF", "PETG"), # global_id=5
  800. ]
  801. )
  802. usage = self._usage([(1, "#CC0000"), (2, "#00CC00")])
  803. assert _match_slots_by_color(usage, {"ams": ams}) == [4, 5]
  804. def test_ams_ht_global_id(self):
  805. """AMS-HT (ams_id >= 128) uses raw ams_id as global tray ID."""
  806. ams = self._ams(
  807. [
  808. (0, 0, "FF0000FF", "PLA"),
  809. (128, 0, "0000FFFF", "PLA"), # AMS-HT → global_id=128
  810. ]
  811. )
  812. usage = self._usage([(1, "#FF0000"), (2, "#0000FF")])
  813. assert _match_slots_by_color(usage, {"ams": ams}) == [0, 128]
  814. def test_ambiguous_same_color_returns_none(self):
  815. """Two trays with the same color → ambiguous → None."""
  816. ams = self._ams(
  817. [
  818. (0, 0, "FF0000FF", "PLA"),
  819. (0, 1, "FF0000FF", "PLA"), # Same red
  820. ]
  821. )
  822. usage = self._usage([(1, "#FF0000")])
  823. assert _match_slots_by_color(usage, {"ams": ams}) is None
  824. def test_no_matching_color_returns_none(self):
  825. """3MF slot color not found in any AMS tray → None."""
  826. ams = self._ams([(0, 0, "00FF00FF", "PLA")])
  827. usage = self._usage([(1, "#FF0000")]) # Red, but AMS has green
  828. assert _match_slots_by_color(usage, {"ams": ams}) is None
  829. def test_color_normalization_strips_alpha(self):
  830. """AMS colors (RRGGBBAA) and 3MF colors (#RRGGBB) match after normalization."""
  831. ams = self._ams([(0, 0, "AABBCC80", "PLA")]) # 8-char with alpha
  832. usage = self._usage([(1, "#AABBCC")]) # 6-char with #
  833. assert _match_slots_by_color(usage, {"ams": ams}) == [0]
  834. def test_case_insensitive(self):
  835. """Color matching is case-insensitive."""
  836. ams = self._ams([(0, 0, "aaBBccFF", "PLA")])
  837. usage = self._usage([(1, "#AAbbCC")])
  838. assert _match_slots_by_color(usage, {"ams": ams}) == [0]
  839. def test_empty_tray_color_skipped(self):
  840. """Trays with empty color are skipped (not matched)."""
  841. ams = self._ams(
  842. [
  843. (0, 0, "", "PLA"),
  844. (0, 1, "FF0000FF", "PLA"),
  845. ]
  846. )
  847. usage = self._usage([(1, "#FF0000")])
  848. assert _match_slots_by_color(usage, {"ams": ams}) == [1]
  849. def test_empty_tray_type_skipped(self):
  850. """Trays with empty tray_type are skipped (unloaded slot)."""
  851. ams = self._ams(
  852. [
  853. (0, 0, "FF0000FF", ""), # Empty slot
  854. (0, 1, "FF0000FF", "PLA"), # Loaded slot
  855. ]
  856. )
  857. usage = self._usage([(1, "#FF0000")])
  858. assert _match_slots_by_color(usage, {"ams": ams}) == [1]
  859. def test_short_slot_color_returns_none(self):
  860. """3MF slot with color < 6 chars → can't match → None."""
  861. ams = self._ams([(0, 0, "FF0000FF", "PLA")])
  862. usage = [{"slot_id": 1, "used_g": 10.0, "type": "PLA", "color": "#FFF"}]
  863. assert _match_slots_by_color(usage, {"ams": ams}) is None
  864. def test_slot_id_zero_skipped(self):
  865. """Slots with slot_id=0 are skipped."""
  866. ams = self._ams([(0, 0, "FF0000FF", "PLA")])
  867. usage = [{"slot_id": 0, "used_g": 10.0, "type": "PLA", "color": "#FF0000"}]
  868. assert _match_slots_by_color(usage, {"ams": ams}) is None
  869. def test_ams_data_as_list(self):
  870. """Handles ams_raw as a plain list (some printer models)."""
  871. ams_list = [{"id": 0, "tray": [{"id": 0, "tray_color": "FF0000FF", "tray_type": "PLA"}]}]
  872. usage = self._usage([(1, "#FF0000")])
  873. assert _match_slots_by_color(usage, ams_list) == [0]
  874. def test_same_color_two_trays_disambiguated_by_usage(self):
  875. """Two trays same color, two slots same color → unique assignment via used_trays tracking."""
  876. ams = self._ams(
  877. [
  878. (0, 0, "FF0000FF", "PLA"),
  879. (0, 1, "FF0000FF", "PLA"),
  880. ]
  881. )
  882. # Two slots both wanting red — first gets tray 0, second gets tray 1? No.
  883. # When first slot takes the only available, second has 1 left → should work
  884. usage = self._usage([(1, "#FF0000"), (2, "#FF0000")])
  885. # First slot: candidates=[0,1], available=[0,1], len!=1 → None
  886. assert _match_slots_by_color(usage, {"ams": ams}) is None
  887. def test_dict_wrapper_with_ams_key(self):
  888. """Standard dict format with 'ams' key."""
  889. ams_data = {"ams": [{"id": 0, "tray": [{"id": 0, "tray_color": "00FF00FF", "tray_type": "PLA"}]}]}
  890. usage = self._usage([(1, "#00FF00")])
  891. assert _match_slots_by_color(usage, ams_data) == [0]
  892. class TestMqttMappingIntegration:
  893. """Integration tests: MQTT mapping field used in _track_from_3mf."""
  894. @pytest.mark.asyncio
  895. async def test_h2c_multi_filament_uses_mqtt_mapping(self):
  896. """H2C: 3 filaments resolved via MQTT mapping field (no ams_mapping, no queue)."""
  897. # AMS0-T1 (White PLA), AMS0-T0 (Black PLA), AMS128-T0 (Red PLA)
  898. spool_white = _make_spool(spool_id=1, label_weight=1000)
  899. spool_black = _make_spool(spool_id=2, label_weight=1000)
  900. spool_red = _make_spool(spool_id=3, label_weight=1000)
  901. assign_white = _make_assignment(spool_id=1, ams_id=0, tray_id=1)
  902. assign_black = _make_assignment(spool_id=2, ams_id=0, tray_id=0)
  903. assign_red = _make_assignment(spool_id=3, ams_id=128, tray_id=0)
  904. archive = _make_archive(archive_id=12)
  905. # db: archive, then 3 pairs of (assignment, spool)
  906. # No queue lookup because MQTT mapping is found first
  907. db = _mock_db_sequential(
  908. [
  909. archive,
  910. assign_white,
  911. spool_white,
  912. assign_black,
  913. spool_black,
  914. assign_red,
  915. spool_red,
  916. ]
  917. )
  918. # MQTT mapping: slot0→AMS0-T1(1), slot1→AMS0-T0(0), slots2-5→unmapped, slot6→AMS128-T0(32768)
  919. printer_manager = MagicMock()
  920. printer_manager.get_status.return_value = SimpleNamespace(
  921. raw_data={"mapping": [1, 0, 65535, 65535, 65535, 65535, 32768]},
  922. progress=100,
  923. layer_num=50,
  924. tray_now=255,
  925. )
  926. # 3MF slots 1, 2, 7 (1-based) → indices 0, 1, 6 in mapping
  927. filament_usage = [
  928. {"slot_id": 1, "used_g": 21.16, "type": "PLA", "color": "#FFFFFF"},
  929. {"slot_id": 2, "used_g": 24.22, "type": "PLA", "color": "#000000"},
  930. {"slot_id": 7, "used_g": 18.47, "type": "PLA", "color": "#F72323"},
  931. ]
  932. handled_trays: set[tuple[int, int]] = set()
  933. with (
  934. patch("backend.app.core.config.settings") as mock_settings,
  935. patch(
  936. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  937. return_value=filament_usage,
  938. ),
  939. ):
  940. mock_settings.base_dir = MagicMock()
  941. mock_path = MagicMock()
  942. mock_path.exists.return_value = True
  943. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  944. results = await _track_from_3mf(
  945. printer_id=1,
  946. archive_id=12,
  947. status="completed",
  948. print_name="Cube + Cube + Cube",
  949. handled_trays=handled_trays,
  950. printer_manager=printer_manager,
  951. db=db,
  952. )
  953. assert len(results) == 3
  954. # slot_id=1 → mapping[0]=1 → AMS0-T1 (White PLA)
  955. assert results[0]["spool_id"] == 1
  956. assert results[0]["ams_id"] == 0
  957. assert results[0]["tray_id"] == 1
  958. assert results[0]["weight_used"] == 21.2
  959. # slot_id=2 → mapping[1]=0 → AMS0-T0 (Black PLA)
  960. assert results[1]["spool_id"] == 2
  961. assert results[1]["ams_id"] == 0
  962. assert results[1]["tray_id"] == 0
  963. assert results[1]["weight_used"] == 24.2
  964. # slot_id=7 → mapping[6]=32768 → AMS128-T0 (Red PLA)
  965. assert results[2]["spool_id"] == 3
  966. assert results[2]["ams_id"] == 128
  967. assert results[2]["tray_id"] == 0
  968. assert results[2]["weight_used"] == 18.5
  969. @pytest.mark.asyncio
  970. async def test_print_cmd_mapping_takes_priority_over_mqtt(self):
  971. """ams_mapping from print command is used even when MQTT mapping exists."""
  972. spool = _make_spool(spool_id=1, label_weight=1000)
  973. assignment = _make_assignment(spool_id=1, ams_id=0, tray_id=2)
  974. archive = _make_archive(archive_id=10)
  975. # db: archive, assignment, spool (no queue lookup when ams_mapping provided)
  976. db = _mock_db_sequential([archive, assignment, spool])
  977. printer_manager = MagicMock()
  978. printer_manager.get_status.return_value = SimpleNamespace(
  979. raw_data={"mapping": [0, 65535]}, # MQTT says slot 0 → AMS0-T0
  980. progress=100,
  981. layer_num=50,
  982. tray_now=255,
  983. )
  984. filament_usage = [{"slot_id": 1, "used_g": 10.0, "type": "PLA", "color": ""}]
  985. handled_trays: set[tuple[int, int]] = set()
  986. with (
  987. patch("backend.app.core.config.settings") as mock_settings,
  988. patch(
  989. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  990. return_value=filament_usage,
  991. ),
  992. ):
  993. mock_settings.base_dir = MagicMock()
  994. mock_path = MagicMock()
  995. mock_path.exists.return_value = True
  996. mock_settings.base_dir.__truediv__ = MagicMock(return_value=mock_path)
  997. results = await _track_from_3mf(
  998. printer_id=1,
  999. archive_id=10,
  1000. status="completed",
  1001. print_name="Test",
  1002. handled_trays=handled_trays,
  1003. printer_manager=printer_manager,
  1004. db=db,
  1005. ams_mapping=[2], # Print cmd says slot 0 → AMS0-T2 (overrides MQTT)
  1006. )
  1007. assert len(results) == 1
  1008. assert results[0]["ams_id"] == 0
  1009. assert results[0]["tray_id"] == 2 # From print_cmd mapping, not MQTT
  1010. class TestNotificationVariables:
  1011. """Tests for filament_details formatting in notifications."""
  1012. def test_filament_details_single_slot(self):
  1013. """Single slot produces 'PLA: 15.2g' format."""
  1014. slots = [{"type": "PLA", "used_g": 15.2, "slot_id": 1, "color": "#FF0000"}]
  1015. parts = []
  1016. for slot in slots:
  1017. ftype = slot.get("type", "Unknown") or "Unknown"
  1018. used = slot.get("used_g", 0)
  1019. parts.append(f"{ftype}: {used:.1f}g")
  1020. result = " | ".join(parts)
  1021. assert result == "PLA: 15.2g"
  1022. def test_filament_details_multi_slot(self):
  1023. """Multiple slots produce 'PLA: 10.0g | PETG: 5.0g' format."""
  1024. slots = [
  1025. {"type": "PLA", "used_g": 10.0, "slot_id": 1, "color": ""},
  1026. {"type": "PETG", "used_g": 5.0, "slot_id": 2, "color": ""},
  1027. ]
  1028. parts = []
  1029. for slot in slots:
  1030. ftype = slot.get("type", "Unknown") or "Unknown"
  1031. used = slot.get("used_g", 0)
  1032. parts.append(f"{ftype}: {used:.1f}g")
  1033. result = " | ".join(parts)
  1034. assert result == "PLA: 10.0g | PETG: 5.0g"
  1035. def test_filament_details_empty_type(self):
  1036. """Empty type defaults to 'Unknown'."""
  1037. slots = [{"type": "", "used_g": 5.0, "slot_id": 1, "color": ""}]
  1038. parts = []
  1039. for slot in slots:
  1040. ftype = slot.get("type", "Unknown") or "Unknown"
  1041. used = slot.get("used_g", 0)
  1042. parts.append(f"{ftype}: {used:.1f}g")
  1043. result = " | ".join(parts)
  1044. assert result == "Unknown: 5.0g"
  1045. def test_filament_grams_scaled_for_partial(self):
  1046. """filament_grams is scaled by progress for partial prints."""
  1047. filament_used_grams = 20.0
  1048. progress = 50
  1049. scale = max(0.0, min(progress / 100.0, 1.0))
  1050. scaled = round(filament_used_grams * scale, 1)
  1051. assert scaled == 10.0
  1052. def test_filament_grams_zero_progress(self):
  1053. """Progress=0 at cancellation gives 0.0g."""
  1054. filament_used_grams = 20.0
  1055. progress = 0
  1056. scale = max(0.0, min(progress / 100.0, 1.0))
  1057. scaled = round(filament_used_grams * scale, 1)
  1058. assert scaled == 0.0
  1059. def test_slot_scaling_for_partial(self):
  1060. """Per-slot usage is scaled linearly for partial prints."""
  1061. slots = [
  1062. {"type": "PLA", "used_g": 20.0, "slot_id": 1, "color": ""},
  1063. {"type": "PETG", "used_g": 10.0, "slot_id": 2, "color": ""},
  1064. ]
  1065. progress = 30
  1066. scale = max(0.0, min(progress / 100.0, 1.0))
  1067. scaled_slots = [{**s, "used_g": round(s["used_g"] * scale, 1)} for s in slots]
  1068. assert scaled_slots[0]["used_g"] == 6.0
  1069. assert scaled_slots[1]["used_g"] == 3.0