test_spoolman_tracking.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  1. """Unit tests for Spoolman tracking service helpers."""
  2. from types import SimpleNamespace
  3. from unittest.mock import AsyncMock, MagicMock, patch
  4. import pytest
  5. from backend.app.services.spoolman_tracking import (
  6. _get_fallback_spool_tag,
  7. _global_tray_id_to_ams_slot,
  8. _hash_serial_to_hex32,
  9. _resolve_global_tray_id,
  10. _resolve_spool_tag,
  11. build_ams_tray_lookup,
  12. store_print_data,
  13. )
  14. class TestResolveSpoolTag:
  15. """Tests for _resolve_spool_tag()."""
  16. def test_prefers_tray_uuid_over_tag_uid(self):
  17. tray = {"tray_uuid": "A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4", "tag_uid": "DEADBEEF"}
  18. assert _resolve_spool_tag(tray) == "A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4"
  19. def test_falls_back_to_tag_uid_when_no_uuid(self):
  20. tray = {"tray_uuid": "", "tag_uid": "DEADBEEF"}
  21. assert _resolve_spool_tag(tray) == "DEADBEEF"
  22. def test_falls_back_to_tag_uid_when_uuid_zero(self):
  23. tray = {"tray_uuid": "00000000000000000000000000000000", "tag_uid": "DEADBEEF"}
  24. assert _resolve_spool_tag(tray) == "DEADBEEF"
  25. def test_rejects_zero_tag_uid(self):
  26. tray = {"tray_uuid": "", "tag_uid": "0000000000000000"}
  27. assert _resolve_spool_tag(tray) == ""
  28. def test_uses_fallback_tag_when_ids_missing(self):
  29. tray = {"tray_uuid": "", "tag_uid": ""}
  30. # global_tray_id 0 -> ams_id 0, tray_id 0
  31. assert _resolve_spool_tag(tray, "01P00A000000000", 0) == "ABA7845700000000"
  32. def test_uses_fallback_tag_when_ids_zero(self):
  33. tray = {"tray_uuid": "00000000000000000000000000000000", "tag_uid": "0000000000000000"}
  34. # global_tray_id 5 -> ams_id 1, tray_id 1
  35. assert _resolve_spool_tag(tray, "01P00A000000000", 5) == "ABA7845700010001"
  36. def test_prefers_tray_uuid_over_fallback_when_non_zero(self):
  37. tray = {"tray_uuid": "A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4", "tag_uid": ""}
  38. assert _resolve_spool_tag(tray, "01P00A000000000", 0) == "A1B2C3D4E5F6A1B2C3D4E5F6A1B2C3D4"
  39. def test_empty_both(self):
  40. tray = {"tray_uuid": "", "tag_uid": ""}
  41. assert _resolve_spool_tag(tray) == ""
  42. def test_missing_keys(self):
  43. assert _resolve_spool_tag({}) == ""
  44. def test_zero_uuid_no_tag(self):
  45. tray = {"tray_uuid": "00000000000000000000000000000000", "tag_uid": ""}
  46. assert _resolve_spool_tag(tray) == ""
  47. class TestResolveGlobalTrayId:
  48. """Tests for _resolve_global_tray_id()."""
  49. def test_default_mapping(self):
  50. """slot 1 -> tray 0, slot 2 -> tray 1, etc."""
  51. assert _resolve_global_tray_id(1, None) == 0
  52. assert _resolve_global_tray_id(2, None) == 1
  53. assert _resolve_global_tray_id(4, None) == 3
  54. def test_custom_mapping(self):
  55. """Custom slot_to_tray overrides default."""
  56. mapping = [5, 2, -1, 0]
  57. assert _resolve_global_tray_id(1, mapping) == 5
  58. assert _resolve_global_tray_id(2, mapping) == 2
  59. assert _resolve_global_tray_id(4, mapping) == 0
  60. def test_unmapped_slot(self):
  61. """Slot with -1 in mapping uses default."""
  62. mapping = [5, -1, 2, 0]
  63. assert _resolve_global_tray_id(2, mapping) == 1 # default: slot 2 -> tray 1
  64. def test_slot_beyond_mapping(self):
  65. """Slot beyond mapping length uses default."""
  66. mapping = [5, 2]
  67. assert _resolve_global_tray_id(3, mapping) == 2 # default: slot 3 -> tray 2
  68. def test_empty_mapping(self):
  69. mapping = []
  70. assert _resolve_global_tray_id(1, mapping) == 0
  71. class TestFallbackTagHelpers:
  72. """Tests for frontend-mirrored fallback tag helpers."""
  73. def test_hash_serial_matches_frontend_algorithm(self):
  74. assert _hash_serial_to_hex32("01P00A000000000") == "ABA78457"
  75. # Frontend trims and uppercases before hashing
  76. assert _hash_serial_to_hex32(" 01p00a000000000 ") == "ABA78457"
  77. def test_global_tray_to_ams_slot_standard_ams(self):
  78. assert _global_tray_id_to_ams_slot(0) == (0, 0)
  79. assert _global_tray_id_to_ams_slot(7) == (1, 3)
  80. def test_global_tray_to_ams_slot_ams_ht(self):
  81. assert _global_tray_id_to_ams_slot(128) == (128, 0)
  82. assert _global_tray_id_to_ams_slot(135) == (135, 0)
  83. def test_global_tray_to_ams_slot_external(self):
  84. assert _global_tray_id_to_ams_slot(254) == (255, 0)
  85. assert _global_tray_id_to_ams_slot(255) == (255, 1)
  86. def test_get_fallback_spool_tag_standard(self):
  87. assert _get_fallback_spool_tag("01P00A000000000", 5) == "ABA7845700010001"
  88. def test_get_fallback_spool_tag_ams_ht(self):
  89. assert _get_fallback_spool_tag("01P00A000000000", 128) == "ABA7845700800000"
  90. def test_get_fallback_spool_tag_external(self):
  91. assert _get_fallback_spool_tag("01P00A000000000", 255) == "ABA7845700FF0001"
  92. class TestBuildAmsTrayLookup:
  93. """Tests for build_ams_tray_lookup()."""
  94. def test_single_ams_unit(self):
  95. raw = {
  96. "ams": [
  97. {
  98. "id": 0,
  99. "tray": [
  100. {"id": 0, "tray_uuid": "AAA", "tag_uid": "111", "tray_type": "PLA"},
  101. {"id": 1, "tray_uuid": "BBB", "tag_uid": "222", "tray_type": "ABS"},
  102. ],
  103. }
  104. ]
  105. }
  106. lookup = build_ams_tray_lookup(raw)
  107. assert lookup[0] == {"tray_uuid": "AAA", "tag_uid": "111", "tray_type": "PLA"}
  108. assert lookup[1] == {"tray_uuid": "BBB", "tag_uid": "222", "tray_type": "ABS"}
  109. def test_multiple_ams_units(self):
  110. raw = {
  111. "ams": [
  112. {"id": 0, "tray": [{"id": 0, "tray_uuid": "A", "tag_uid": "", "tray_type": "PLA"}]},
  113. {"id": 1, "tray": [{"id": 0, "tray_uuid": "B", "tag_uid": "", "tray_type": "PETG"}]},
  114. ]
  115. }
  116. lookup = build_ams_tray_lookup(raw)
  117. assert 0 in lookup # AMS 0, tray 0
  118. assert 4 in lookup # AMS 1, tray 0 (1*4+0)
  119. assert lookup[4]["tray_uuid"] == "B"
  120. def test_external_spool(self):
  121. raw = {
  122. "ams": [],
  123. "vt_tray": [{"tray_uuid": "EXT", "tag_uid": "X", "tray_type": "TPU"}],
  124. }
  125. lookup = build_ams_tray_lookup(raw)
  126. assert 254 in lookup
  127. assert lookup[254]["tray_type"] == "TPU"
  128. def test_empty_external_spool_skipped(self):
  129. raw = {"ams": [], "vt_tray": [{"tray_type": ""}]}
  130. lookup = build_ams_tray_lookup(raw)
  131. assert 254 not in lookup
  132. def test_no_ams_data(self):
  133. assert build_ams_tray_lookup({}) == {}
  134. assert build_ams_tray_lookup({"ams": []}) == {}
  135. def test_missing_fields_default(self):
  136. raw = {"ams": [{"id": 0, "tray": [{"id": 0}]}]}
  137. lookup = build_ams_tray_lookup(raw)
  138. assert lookup[0] == {"tray_uuid": "", "tag_uid": "", "tray_type": ""}
  139. class TestStorePrintData:
  140. """Tests for store_print_data()."""
  141. @pytest.mark.asyncio
  142. async def test_prefers_explicit_ams_mapping_over_queue_mapping(self):
  143. db = AsyncMock()
  144. delete_result = MagicMock()
  145. db.execute = AsyncMock(side_effect=[delete_result])
  146. db.add = MagicMock()
  147. db.commit = AsyncMock()
  148. printer_manager = MagicMock()
  149. printer_manager.get_status.return_value = SimpleNamespace(
  150. raw_data={"ams": [{"id": 0, "tray": [{"id": 0, "tray_type": "PLA"}, {"id": 1, "tray_type": "PLA"}]}]}
  151. )
  152. mock_settings = MagicMock()
  153. mock_path = MagicMock()
  154. mock_path.exists.return_value = True
  155. mock_settings.base_dir.__truediv__.return_value = mock_path
  156. with (
  157. patch("backend.app.services.spoolman_tracking.app_settings", mock_settings),
  158. patch("backend.app.api.routes.settings.get_setting", AsyncMock(side_effect=["true", "true"])),
  159. patch(
  160. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  161. return_value=[{"slot_id": 1, "used_g": 3.83, "type": "PLA", "color": "#FF0000"}],
  162. ),
  163. patch("backend.app.utils.threemf_tools.extract_layer_filament_usage_from_3mf", return_value=None),
  164. patch("backend.app.utils.threemf_tools.extract_filament_properties_from_3mf", return_value={}),
  165. ):
  166. await store_print_data(
  167. printer_id=1,
  168. archive_id=15,
  169. file_path="archives/test.3mf",
  170. db=db,
  171. printer_manager=printer_manager,
  172. ams_mapping=[1, -1, -1, -1],
  173. )
  174. db.add.assert_called_once()
  175. tracking = db.add.call_args.args[0]
  176. assert tracking.slot_to_tray == [1, -1, -1, -1]
  177. db.execute.assert_called_once()
  178. @pytest.mark.asyncio
  179. async def test_stores_tracking_when_disable_weight_sync_is_false(self):
  180. """#1119: per-print tracking must run regardless of disable_weight_sync.
  181. Previously store_print_data short-circuited when the deprecated
  182. `spoolman_disable_weight_sync` flag was off, leaving non-BL spools
  183. with no weight-update path at all. Per-print tracking is now the
  184. only weight writer for Spoolman, so it must run whenever Spoolman
  185. is enabled.
  186. """
  187. db = AsyncMock()
  188. db.execute = AsyncMock(return_value=MagicMock())
  189. db.add = MagicMock()
  190. db.commit = AsyncMock()
  191. printer_manager = MagicMock()
  192. printer_manager.get_status.return_value = SimpleNamespace(
  193. raw_data={"ams": [{"id": 0, "tray": [{"id": 0, "tray_type": "PLA"}]}]}
  194. )
  195. mock_settings = MagicMock()
  196. mock_path = MagicMock()
  197. mock_path.exists.return_value = True
  198. mock_settings.base_dir.__truediv__.return_value = mock_path
  199. # Only spoolman_enabled is consulted now (disable_weight_sync is no
  200. # longer read). The single side_effect entry proves no extra
  201. # get_setting calls slip back in.
  202. with (
  203. patch("backend.app.services.spoolman_tracking.app_settings", mock_settings),
  204. patch("backend.app.api.routes.settings.get_setting", AsyncMock(side_effect=["true"])),
  205. patch(
  206. "backend.app.utils.threemf_tools.extract_filament_usage_from_3mf",
  207. return_value=[{"slot_id": 1, "used_g": 5.0, "type": "PLA", "color": "#FF0000"}],
  208. ),
  209. patch("backend.app.utils.threemf_tools.extract_layer_filament_usage_from_3mf", return_value=None),
  210. patch("backend.app.utils.threemf_tools.extract_filament_properties_from_3mf", return_value={}),
  211. ):
  212. await store_print_data(
  213. printer_id=1,
  214. archive_id=20,
  215. file_path="archives/test.3mf",
  216. db=db,
  217. printer_manager=printer_manager,
  218. ams_mapping=[0],
  219. )
  220. # Tracking row was inserted — the fix is working.
  221. db.add.assert_called_once()