test_layer_timelapse_expected_archive.py 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. """Regression test for #1353: layer timelapse must start for queue/VP-dispatched prints.
  2. Reporter @Andlar94 ran the external-camera flow on an A1 dispatched via the
  3. print queue (so each print landed in the on_print_start "expected archive"
  4. branch). Frames were never captured, no MP4 was produced, yet the post-print
  5. log line said "Stitching layer timelapse for printer 1" — because
  6. `tl_complete()` ran, found no active session, and silently returned None.
  7. Root cause: only the two new-archive code paths in on_print_start
  8. (`fallback_archive` + `archive_print`) called `layer_timelapse.start_session`.
  9. The expected-archive branch — where reprints and queue dispatch land —
  10. updated the existing archive's status to "printing" but never started a
  11. timelapse session.
  12. Fix: start_session is now called in the expected-archive branch too, guarded
  13. by the same `external_camera_enabled and external_camera_url` check that
  14. the other two paths use.
  15. """
  16. from unittest.mock import AsyncMock, MagicMock, patch
  17. import pytest
  18. from backend.app.main import (
  19. _active_prints,
  20. _expected_print_creators,
  21. _expected_print_registered_at,
  22. _expected_prints,
  23. _print_ams_mappings,
  24. register_expected_print,
  25. )
  26. @pytest.fixture(autouse=True)
  27. def _clear_dicts():
  28. """Clear module-level tracking dicts before and after each test."""
  29. _expected_prints.clear()
  30. _expected_print_registered_at.clear()
  31. _expected_print_creators.clear()
  32. _print_ams_mappings.clear()
  33. _active_prints.clear()
  34. yield
  35. _expected_prints.clear()
  36. _expected_print_registered_at.clear()
  37. _expected_print_creators.clear()
  38. _print_ams_mappings.clear()
  39. _active_prints.clear()
  40. def _build_mocks(*, external_camera_enabled: bool, external_camera_url: str | None):
  41. """Construct the mock matrix needed to drive on_print_start through the
  42. expected-archive branch. Returns a dict of mock contexts that the test
  43. enters via contextlib.ExitStack.
  44. The session.execute mock returns the printer for the first call (printer
  45. lookup) and the archive row for the second call (expected-archive
  46. re-fetch). The archive row carries a unique filename so the
  47. expected-print key lookup succeeds.
  48. """
  49. mock_printer = MagicMock()
  50. mock_printer.id = 1
  51. mock_printer.auto_archive = True
  52. mock_printer.external_camera_enabled = external_camera_enabled
  53. mock_printer.external_camera_url = external_camera_url
  54. mock_printer.external_camera_type = "snapshot"
  55. mock_printer.external_camera_snapshot_url = external_camera_url
  56. mock_printer.name = "TestA1"
  57. mock_archive = MagicMock()
  58. mock_archive.id = 42
  59. mock_archive.filename = "Universal_Spirit_level_Holder.3mf"
  60. mock_archive.subtask_id = None
  61. mock_archive.print_time_seconds = None
  62. mock_archive.created_by_id = None
  63. mock_archive.printer_id = 1
  64. mock_archive.print_name = "Universal Spirit Level Holder"
  65. mock_archive.status = "pending"
  66. mock_archive.file_path = "/test/archives/fake.3mf"
  67. return mock_printer, mock_archive
  68. @pytest.mark.asyncio
  69. async def test_expected_archive_path_starts_timelapse_when_external_camera_enabled():
  70. """Queue/VP-dispatched prints land in the expected-archive branch and must
  71. start the timelapse session there (the #1353 root cause)."""
  72. mock_printer, mock_archive = _build_mocks(
  73. external_camera_enabled=True, external_camera_url="http://camera.local:5000/snapshot.jpg"
  74. )
  75. # Register the expected print so the dispatch flow finds an archive_id.
  76. register_expected_print(1, "Universal_Spirit_level_Holder.3mf", archive_id=42, ams_mapping=[1])
  77. # on_print_start fires many db.execute() calls (settings lookups,
  78. # usage tracker, plate detection, etc) before reaching the expected-
  79. # archive branch. Route on SQL text so each query gets a sensible
  80. # response regardless of order, rather than queuing N mocks.
  81. def execute_router(stmt, *args, **kwargs):
  82. sql = str(stmt).lower()
  83. if "from printers" in sql or "from printer " in sql:
  84. return MagicMock(
  85. scalar_one_or_none=MagicMock(return_value=mock_printer),
  86. scalars=MagicMock(return_value=MagicMock(all=MagicMock(return_value=[mock_printer]))),
  87. )
  88. if "from print_archives" in sql or "from print_archive" in sql:
  89. return MagicMock(
  90. scalar_one_or_none=MagicMock(return_value=mock_archive),
  91. scalars=MagicMock(return_value=MagicMock(all=MagicMock(return_value=[mock_archive]))),
  92. )
  93. # Settings, spool assignments, anything else — return empty.
  94. return MagicMock(
  95. scalar_one_or_none=MagicMock(return_value=None),
  96. scalars=MagicMock(return_value=MagicMock(all=MagicMock(return_value=[]))),
  97. )
  98. mock_session = AsyncMock()
  99. mock_session.__aenter__ = AsyncMock(return_value=mock_session)
  100. mock_session.__aexit__ = AsyncMock()
  101. mock_session.execute = AsyncMock(side_effect=execute_router)
  102. mock_session.commit = AsyncMock()
  103. with (
  104. patch("backend.app.main.async_session") as mock_session_maker,
  105. patch("backend.app.main.notification_service") as mock_notif,
  106. patch("backend.app.main.smart_plug_manager") as mock_plug,
  107. patch("backend.app.main.ws_manager") as mock_ws,
  108. patch("backend.app.main.printer_manager") as mock_pm,
  109. patch("backend.app.main.mqtt_relay") as mock_relay,
  110. patch("backend.app.main._record_energy_start", new_callable=AsyncMock),
  111. patch("backend.app.main._load_objects_from_archive"),
  112. patch("backend.app.main._store_spoolman_print_data", new_callable=AsyncMock),
  113. patch("backend.app.main._send_print_start_notification", new_callable=AsyncMock),
  114. # The actual subject under test: assert start_session is called.
  115. patch("backend.app.services.layer_timelapse.start_session") as mock_start_session,
  116. ):
  117. mock_session_maker.return_value = mock_session
  118. mock_notif.on_print_start = AsyncMock()
  119. mock_plug.on_print_start = AsyncMock()
  120. mock_ws.send_print_start = AsyncMock()
  121. mock_ws.send_archive_updated = AsyncMock()
  122. mock_relay.on_print_start = AsyncMock()
  123. mock_pm.get_printer = MagicMock(return_value=MagicMock(name="Test", serial_number="TEST123"))
  124. from backend.app.main import on_print_start
  125. await on_print_start(
  126. 1,
  127. {
  128. "filename": "Universal_Spirit_level_Holder.3mf",
  129. "subtask_name": "Universal_Spirit_level_Holder",
  130. },
  131. )
  132. mock_start_session.assert_called_once()
  133. # Verify it was called with the archive_id from the expected-print
  134. # registration, not a fresh one — that's the contract.
  135. call_args = mock_start_session.call_args
  136. assert call_args.args[0] == 1, "printer_id must match"
  137. assert call_args.args[1] == 42, "archive_id must come from the expected-print registration"
  138. assert call_args.args[2] == "http://camera.local:5000/snapshot.jpg"
  139. assert call_args.args[3] == "snapshot"
  140. @pytest.mark.asyncio
  141. async def test_expected_archive_path_skips_timelapse_when_external_camera_disabled():
  142. """The same guard that the new-archive paths use must hold here: no
  143. external camera → no timelapse session. Otherwise we'd try to capture
  144. from a None URL and crash the print-start flow."""
  145. mock_printer, mock_archive = _build_mocks(external_camera_enabled=False, external_camera_url=None)
  146. mock_archive.filename = "test.3mf"
  147. mock_archive.id = 99
  148. register_expected_print(1, "test.3mf", archive_id=99, ams_mapping=None)
  149. def execute_router(stmt, *args, **kwargs):
  150. sql = str(stmt).lower()
  151. if "from printers" in sql or "from printer " in sql:
  152. return MagicMock(
  153. scalar_one_or_none=MagicMock(return_value=mock_printer),
  154. scalars=MagicMock(return_value=MagicMock(all=MagicMock(return_value=[mock_printer]))),
  155. )
  156. if "from print_archives" in sql or "from print_archive" in sql:
  157. return MagicMock(
  158. scalar_one_or_none=MagicMock(return_value=mock_archive),
  159. scalars=MagicMock(return_value=MagicMock(all=MagicMock(return_value=[mock_archive]))),
  160. )
  161. return MagicMock(
  162. scalar_one_or_none=MagicMock(return_value=None),
  163. scalars=MagicMock(return_value=MagicMock(all=MagicMock(return_value=[]))),
  164. )
  165. mock_session = AsyncMock()
  166. mock_session.__aenter__ = AsyncMock(return_value=mock_session)
  167. mock_session.__aexit__ = AsyncMock()
  168. mock_session.execute = AsyncMock(side_effect=execute_router)
  169. mock_session.commit = AsyncMock()
  170. with (
  171. patch("backend.app.main.async_session") as mock_session_maker,
  172. patch("backend.app.main.notification_service") as mock_notif,
  173. patch("backend.app.main.smart_plug_manager") as mock_plug,
  174. patch("backend.app.main.ws_manager") as mock_ws,
  175. patch("backend.app.main.printer_manager") as mock_pm,
  176. patch("backend.app.main.mqtt_relay") as mock_relay,
  177. patch("backend.app.main._record_energy_start", new_callable=AsyncMock),
  178. patch("backend.app.main._load_objects_from_archive"),
  179. patch("backend.app.main._store_spoolman_print_data", new_callable=AsyncMock),
  180. patch("backend.app.main._send_print_start_notification", new_callable=AsyncMock),
  181. patch("backend.app.services.layer_timelapse.start_session") as mock_start_session,
  182. ):
  183. mock_session_maker.return_value = mock_session
  184. mock_notif.on_print_start = AsyncMock()
  185. mock_plug.on_print_start = AsyncMock()
  186. mock_ws.send_print_start = AsyncMock()
  187. mock_ws.send_archive_updated = AsyncMock()
  188. mock_relay.on_print_start = AsyncMock()
  189. mock_pm.get_printer = MagicMock(return_value=MagicMock(name="Test", serial_number="TEST123"))
  190. from backend.app.main import on_print_start
  191. await on_print_start(1, {"filename": "test.3mf", "subtask_name": "test"})
  192. mock_start_session.assert_not_called()