| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886 |
- """Tests for the auto-drying feature in the print scheduler.
- Covers:
- - Conservative drying parameter selection (mixed filaments)
- - Drying preset loading (user-configured vs defaults)
- - Auto-drying lifecycle: start, humidity stop, minimum drying time
- - Auto-drying stop conditions: feature disabled, no scheduled items, per-printer
- - Sync drying state after restart
- """
- import time
- from unittest.mock import AsyncMock, MagicMock, patch
- import pytest
- from backend.app.services.print_scheduler import PrintScheduler
- class TestConservativeDryingParams:
- """Test _get_conservative_drying_params — picks safest temp/duration for mixed filaments."""
- @pytest.fixture
- def scheduler(self):
- return PrintScheduler()
- def test_single_filament_pla(self, scheduler):
- """Single PLA tray uses PLA preset."""
- trays = [{"tray_type": "PLA"}]
- presets = PrintScheduler.DEFAULT_DRYING_PRESETS
- result = scheduler._get_conservative_drying_params(trays, "n3f", presets)
- assert result == (45, 12, "PLA")
- def test_mixed_filaments_lowest_temp(self, scheduler):
- """Mixed PLA + ABS: should use PLA's 45°C (lowest), ABS's 12h (longest for n3f)."""
- trays = [{"tray_type": "PLA"}, {"tray_type": "ABS"}]
- presets = PrintScheduler.DEFAULT_DRYING_PRESETS
- result = scheduler._get_conservative_drying_params(trays, "n3f", presets)
- temp, hours, _ = result
- assert temp == 45 # PLA is lowest
- assert hours == 12
- def test_mixed_filaments_longest_duration(self, scheduler):
- """Mixed ABS (8h) + PVA (18h) on n3s: should use longest duration."""
- trays = [{"tray_type": "ABS"}, {"tray_type": "PVA"}]
- presets = PrintScheduler.DEFAULT_DRYING_PRESETS
- result = scheduler._get_conservative_drying_params(trays, "n3s", presets)
- temp, hours, _ = result
- assert temp == 80 # ABS n3s=80, PVA n3s=85 → lowest=80
- assert hours == 18 # ABS n3s_hours=8, PVA n3s_hours=18 → longest=18
- def test_empty_trays_returns_none(self, scheduler):
- """No loaded trays returns None."""
- result = scheduler._get_conservative_drying_params([], "n3f", PrintScheduler.DEFAULT_DRYING_PRESETS)
- assert result is None
- def test_unknown_filament_skipped(self, scheduler):
- """Unknown filament types are ignored."""
- trays = [{"tray_type": "EXOTIC_WOOD"}]
- result = scheduler._get_conservative_drying_params(trays, "n3f", PrintScheduler.DEFAULT_DRYING_PRESETS)
- assert result is None
- def test_filament_type_normalization(self, scheduler):
- """'PLA Basic' should normalize to 'PLA'."""
- trays = [{"tray_type": "PLA Basic"}]
- presets = PrintScheduler.DEFAULT_DRYING_PRESETS
- result = scheduler._get_conservative_drying_params(trays, "n3f", presets)
- assert result is not None
- assert result[0] == 45 # PLA temp
- def test_empty_tray_type_skipped(self, scheduler):
- """Trays with empty tray_type are skipped."""
- trays = [{"tray_type": ""}, {"tray_type": "PETG"}]
- presets = PrintScheduler.DEFAULT_DRYING_PRESETS
- result = scheduler._get_conservative_drying_params(trays, "n3f", presets)
- assert result is not None
- assert result[2] == "PETG"
- def test_n3s_uses_n3s_keys(self, scheduler):
- """AMS-HT (n3s) should use n3s temp and n3s_hours."""
- trays = [{"tray_type": "TPU"}]
- presets = PrintScheduler.DEFAULT_DRYING_PRESETS
- result = scheduler._get_conservative_drying_params(trays, "n3s", presets)
- assert result == (75, 18, "TPU") # n3s=75, n3s_hours=18
- def test_n3f_uses_n3f_keys(self, scheduler):
- """AMS 2 Pro (n3f) should use n3f temp and n3f_hours."""
- trays = [{"tray_type": "TPU"}]
- presets = PrintScheduler.DEFAULT_DRYING_PRESETS
- result = scheduler._get_conservative_drying_params(trays, "n3f", presets)
- assert result == (65, 12, "TPU") # n3f=65, n3f_hours=12
- def test_custom_presets(self, scheduler):
- """Custom presets override defaults."""
- trays = [{"tray_type": "PLA"}]
- custom = {"PLA": {"n3f": 50, "n3s": 50, "n3f_hours": 6, "n3s_hours": 6}}
- result = scheduler._get_conservative_drying_params(trays, "n3f", custom)
- assert result == (50, 6, "PLA")
- class TestDryingPresets:
- """Test _get_drying_presets — loads user presets from DB or falls back to defaults."""
- @pytest.fixture
- def scheduler(self):
- return PrintScheduler()
- @pytest.mark.asyncio
- async def test_default_presets_when_no_setting(self, scheduler):
- """Returns built-in defaults when no DB setting exists."""
- db = AsyncMock()
- result_mock = MagicMock()
- result_mock.scalar_one_or_none.return_value = None
- db.execute = AsyncMock(return_value=result_mock)
- presets = await scheduler._get_drying_presets(db)
- assert presets == PrintScheduler.DEFAULT_DRYING_PRESETS
- @pytest.mark.asyncio
- async def test_user_presets_from_db(self, scheduler):
- """Returns user-configured presets when saved in DB."""
- db = AsyncMock()
- setting = MagicMock()
- setting.value = '{"PLA": {"n3f": 50, "n3s": 50, "n3f_hours": 6, "n3s_hours": 6}}'
- result_mock = MagicMock()
- result_mock.scalar_one_or_none.return_value = setting
- db.execute = AsyncMock(return_value=result_mock)
- presets = await scheduler._get_drying_presets(db)
- assert presets["PLA"]["n3f"] == 50
- @pytest.mark.asyncio
- async def test_invalid_json_falls_back(self, scheduler):
- """Invalid JSON in DB falls back to defaults."""
- db = AsyncMock()
- setting = MagicMock()
- setting.value = "not valid json{{"
- result_mock = MagicMock()
- result_mock.scalar_one_or_none.return_value = setting
- db.execute = AsyncMock(return_value=result_mock)
- presets = await scheduler._get_drying_presets(db)
- assert presets == PrintScheduler.DEFAULT_DRYING_PRESETS
- @pytest.mark.asyncio
- async def test_empty_string_falls_back(self, scheduler):
- """Empty string in DB falls back to defaults."""
- db = AsyncMock()
- setting = MagicMock()
- setting.value = ""
- result_mock = MagicMock()
- result_mock.scalar_one_or_none.return_value = setting
- db.execute = AsyncMock(return_value=result_mock)
- presets = await scheduler._get_drying_presets(db)
- assert presets == PrintScheduler.DEFAULT_DRYING_PRESETS
- class TestSyncDryingState:
- """Test _sync_drying_state — syncs in-memory state with actual printer status."""
- @pytest.fixture
- def scheduler(self):
- return PrintScheduler()
- @patch("backend.app.services.print_scheduler.printer_manager")
- def test_removes_stopped_printers(self, mock_pm, scheduler):
- """Printers that stopped drying are removed from tracking."""
- scheduler._drying_in_progress = {1: time.monotonic()}
- state = MagicMock()
- state.raw_data = {"ams": [{"dry_time": 0}]}
- mock_pm.get_status.return_value = state
- scheduler._sync_drying_state()
- assert 1 not in scheduler._drying_in_progress
- @patch("backend.app.services.print_scheduler.printer_manager")
- def test_keeps_active_printers(self, mock_pm, scheduler):
- """Printers still drying remain in tracking."""
- ts = time.monotonic()
- scheduler._drying_in_progress = {1: ts}
- state = MagicMock()
- state.raw_data = {"ams": [{"dry_time": 120}]}
- mock_pm.get_status.return_value = state
- scheduler._sync_drying_state()
- assert scheduler._drying_in_progress[1] == ts
- @patch("backend.app.services.print_scheduler.printer_manager")
- def test_removes_disconnected_printers(self, mock_pm, scheduler):
- """Disconnected printers are removed from tracking."""
- scheduler._drying_in_progress = {1: time.monotonic()}
- mock_pm.get_status.return_value = None
- scheduler._sync_drying_state()
- assert 1 not in scheduler._drying_in_progress
- class TestStopDrying:
- """Test _stop_drying — sends stop commands and clears tracking."""
- @pytest.fixture
- def scheduler(self):
- return PrintScheduler()
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- async def test_stops_all_ams_units(self, mock_pm, scheduler):
- """Sends stop command to each AMS unit that is drying."""
- scheduler._drying_in_progress = {1: time.monotonic()}
- state = MagicMock()
- state.raw_data = {
- "ams": [
- {"id": 0, "dry_time": 120},
- {"id": 1, "dry_time": 0},
- {"id": 128, "dry_time": 60},
- ]
- }
- mock_pm.get_status.return_value = state
- await scheduler._stop_drying(1)
- # Should send stop to AMS 0 and 128, not AMS 1
- calls = mock_pm.send_drying_command.call_args_list
- assert len(calls) == 2
- assert calls[0].args == (1, 0, 0, 0)
- assert calls[1].args == (1, 128, 0, 0)
- assert 1 not in scheduler._drying_in_progress
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- async def test_clears_tracking_when_no_state(self, mock_pm, scheduler):
- """Clears tracking when printer has no state (disconnected)."""
- scheduler._drying_in_progress = {1: time.monotonic()}
- mock_pm.get_status.return_value = None
- await scheduler._stop_drying(1)
- assert 1 not in scheduler._drying_in_progress
- class TestMinimumDryingTime:
- """Regression: drying should not stop/restart rapidly when humidity oscillates near threshold."""
- @pytest.fixture
- def scheduler(self):
- s = PrintScheduler()
- s._min_drying_seconds = 1800 # 30 minutes
- return s
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- @patch("backend.app.services.print_scheduler.supports_drying", return_value=True)
- async def test_no_stop_before_minimum_time(self, mock_sd, mock_pm, scheduler):
- """Drying should NOT stop when humidity drops below threshold before 30 min."""
- # Simulate: drying started 5 minutes ago
- scheduler._drying_in_progress = {1: time.monotonic() - 300}
- state = MagicMock()
- state.raw_data = {
- "ams": [
- {
- "id": 0,
- "module_type": "n3f",
- "dry_time": 600,
- "humidity_raw": "18",
- "dry_sf_reason": [],
- "tray": [{"tray_type": "PLA"}],
- }
- ]
- }
- state.firmware_version = "01.09.00.00"
- mock_pm.get_status.return_value = state
- mock_pm.is_connected.return_value = True
- mock_pm.get_model.return_value = "X1C"
- # Mock _is_printer_idle and DB
- scheduler._is_printer_idle = MagicMock(return_value=True)
- db = AsyncMock()
- # Mock settings: enabled, threshold=21
- settings_returns = {
- "queue_drying_enabled": self._make_setting("true"),
- "ams_humidity_fair": self._make_setting("21"),
- "queue_drying_block": self._make_setting("false"),
- "drying_presets": None,
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns, printer_id=1))
- # Queue item with schedule
- item = MagicMock()
- item.printer_id = 1
- item.scheduled_time = MagicMock() # Has a schedule
- item.manual_start = False
- await scheduler._check_auto_drying(db, [item], set())
- # Should NOT have sent stop command via humidity check — minimum time not elapsed
- # The only calls should NOT include the humidity-based stop
- for call in mock_pm.send_drying_command.call_args_list:
- # If any stop was called, it should NOT be from the humidity path
- # (humidity path uses keyword args: temp=0, duration=0, mode=0)
- assert call != ((1, 0), {"temp": 0, "duration": 0, "mode": 0}), (
- "Humidity-based stop should not fire before minimum drying time"
- )
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- @patch("backend.app.services.print_scheduler.supports_drying", return_value=True)
- async def test_stops_after_minimum_time(self, mock_sd, mock_pm, scheduler):
- """Drying SHOULD stop when humidity below threshold AND 30 min elapsed."""
- # Simulate: drying started 35 minutes ago
- scheduler._drying_in_progress = {1: time.monotonic() - 2100}
- state = MagicMock()
- state.raw_data = {
- "ams": [
- {
- "id": 0,
- "module_type": "n3f",
- "dry_time": 600,
- "humidity_raw": "18",
- "dry_sf_reason": [],
- "tray": [{"tray_type": "PLA"}],
- }
- ]
- }
- state.firmware_version = "01.09.00.00"
- mock_pm.get_status.return_value = state
- mock_pm.is_connected.return_value = True
- mock_pm.get_model.return_value = "X1C"
- scheduler._is_printer_idle = MagicMock(return_value=True)
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("true"),
- "ams_humidity_fair": self._make_setting("21"),
- "queue_drying_block": self._make_setting("false"),
- "drying_presets": None,
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns, printer_id=1))
- item = MagicMock()
- item.printer_id = 1
- item.scheduled_time = MagicMock()
- item.manual_start = False
- await scheduler._check_auto_drying(db, [item], set())
- # Should have sent stop command (humidity-based stop after minimum time)
- mock_pm.send_drying_command.assert_any_call(1, 0, temp=0, duration=0, mode=0)
- @staticmethod
- def _make_setting(value):
- s = MagicMock()
- s.value = value
- return s
- @staticmethod
- def _make_db_side_effect(settings_map, printer_id=1):
- """Create a side_effect for db.execute that returns settings and printers."""
- async def side_effect(stmt):
- result = MagicMock()
- stmt_str = str(stmt)
- # Extract bind parameter values (SQLAlchemy uses :key_1 placeholders)
- try:
- compiled = stmt.compile(compile_kwargs={"literal_binds": False})
- param_values = list(compiled.params.values())
- except Exception:
- param_values = []
- # Match settings queries by checking bind parameter values
- matched = False
- for key, val in settings_map.items():
- if key in param_values:
- result.scalar_one_or_none.return_value = val
- matched = True
- break
- if not matched:
- if "printer" in stmt_str.lower() or "is_active" in stmt_str:
- printer = MagicMock()
- printer.id = printer_id
- printer.is_active = True
- scalars_mock = MagicMock()
- scalars_mock.__iter__ = MagicMock(return_value=iter([printer]))
- result.scalars.return_value = scalars_mock
- else:
- result.scalar_one_or_none.return_value = None
- return result
- return side_effect
- class TestAutoStopOnFeatureDisabled:
- """Regression: disabling auto-drying in settings should stop active drying sessions."""
- @pytest.fixture
- def scheduler(self):
- return PrintScheduler()
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- async def test_stops_drying_when_disabled(self, mock_pm, scheduler):
- """Disabling auto-drying should send stop commands to all drying printers."""
- scheduler._drying_in_progress = {1: time.monotonic(), 2: time.monotonic()}
- # Printer 1: drying, Printer 2: drying
- def get_status(pid):
- state = MagicMock()
- state.raw_data = {"ams": [{"id": 0, "dry_time": 120}]}
- return state
- mock_pm.get_status.side_effect = get_status
- db = AsyncMock()
- # queue_drying_enabled = false
- setting = MagicMock()
- setting.value = "false"
- result_mock = MagicMock()
- result_mock.scalar_one_or_none.return_value = setting
- db.execute = AsyncMock(return_value=result_mock)
- await scheduler._check_auto_drying(db, [], set())
- # Should have sent stop commands
- assert mock_pm.send_drying_command.call_count == 2
- assert not scheduler._drying_in_progress
- class TestAutoStopOnNoScheduledItems:
- """Regression: removing scheduled items should stop auto-drying."""
- @pytest.fixture
- def scheduler(self):
- return PrintScheduler()
- @staticmethod
- def _make_setting(value):
- s = MagicMock()
- s.value = value
- return s
- @staticmethod
- def _make_db_side_effect(settings_map):
- """Create a side_effect for db.execute that returns settings by key."""
- async def side_effect(stmt):
- result = MagicMock()
- try:
- compiled = stmt.compile(compile_kwargs={"literal_binds": False})
- param_values = list(compiled.params.values())
- except Exception:
- param_values = []
- for key, val in settings_map.items():
- if key in param_values:
- result.scalar_one_or_none.return_value = val
- return result
- result.scalar_one_or_none.return_value = None
- return result
- return side_effect
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- async def test_stops_when_no_scheduled_items(self, mock_pm, scheduler):
- """Auto-drying stops when queue has no scheduled items (queue mode only)."""
- scheduler._drying_in_progress = {1: time.monotonic()}
- state = MagicMock()
- state.raw_data = {"ams": [{"id": 0, "dry_time": 120}]}
- mock_pm.get_status.return_value = state
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("true"),
- "ambient_drying_enabled": self._make_setting("false"),
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns))
- # Manual-start items only (no scheduled_time)
- item = MagicMock()
- item.printer_id = 1
- item.scheduled_time = None
- item.manual_start = True
- await scheduler._check_auto_drying(db, [item], set())
- # Should have stopped drying
- assert mock_pm.send_drying_command.called
- assert not scheduler._drying_in_progress
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- async def test_stops_when_empty_queue(self, mock_pm, scheduler):
- """Auto-drying stops when queue is completely empty (queue mode only)."""
- scheduler._drying_in_progress = {1: time.monotonic()}
- state = MagicMock()
- state.raw_data = {"ams": [{"id": 0, "dry_time": 120}]}
- mock_pm.get_status.return_value = state
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("true"),
- "ambient_drying_enabled": self._make_setting("false"),
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns))
- await scheduler._check_auto_drying(db, [], set())
- assert mock_pm.send_drying_command.called
- assert not scheduler._drying_in_progress
- class TestDryingTrackingTimestamps:
- """Test that _drying_in_progress uses timestamps, not booleans."""
- def test_initial_state_empty(self):
- """Fresh scheduler has no drying tracked."""
- scheduler = PrintScheduler()
- assert scheduler._drying_in_progress == {}
- def test_timestamp_is_monotonic(self):
- """Tracked values should be monotonic timestamps."""
- scheduler = PrintScheduler()
- before = time.monotonic()
- scheduler._drying_in_progress[1] = time.monotonic()
- after = time.monotonic()
- assert before <= scheduler._drying_in_progress[1] <= after
- def test_timestamp_is_truthy(self):
- """Timestamps are truthy for .get() checks (backward compat with bool pattern)."""
- scheduler = PrintScheduler()
- scheduler._drying_in_progress[1] = time.monotonic()
- assert scheduler._drying_in_progress.get(1)
- assert not scheduler._drying_in_progress.get(999)
- class _DryingTestBase:
- """Shared helpers for auto-drying integration tests."""
- @staticmethod
- def _make_setting(value):
- s = MagicMock()
- s.value = value
- return s
- @staticmethod
- def _make_db_side_effect(settings_map, printer_ids=None):
- """Create a side_effect for db.execute that returns settings by key and printers."""
- if printer_ids is None:
- printer_ids = [1]
- async def side_effect(stmt):
- result = MagicMock()
- stmt_str = str(stmt)
- try:
- compiled = stmt.compile(compile_kwargs={"literal_binds": False})
- param_values = list(compiled.params.values())
- except Exception:
- param_values = []
- for key, val in settings_map.items():
- if key in param_values:
- result.scalar_one_or_none.return_value = val
- return result
- if "printer" in stmt_str.lower() or "is_active" in stmt_str:
- printers = []
- for pid in printer_ids:
- p = MagicMock()
- p.id = pid
- p.is_active = True
- printers.append(p)
- scalars_mock = MagicMock()
- scalars_mock.__iter__ = MagicMock(return_value=iter(printers))
- result.scalars.return_value = scalars_mock
- else:
- result.scalar_one_or_none.return_value = None
- return result
- return side_effect
- class TestAmbientDrying(_DryingTestBase):
- """Tests for ambient drying mode — drying based on humidity regardless of queue state."""
- @pytest.fixture
- def scheduler(self):
- return PrintScheduler()
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- @patch("backend.app.services.print_scheduler.supports_drying", return_value=True)
- async def test_ambient_dries_idle_printer_without_queue(self, mock_sd, mock_pm, scheduler):
- """Ambient mode starts drying on idle printers even with no queue items."""
- state = MagicMock()
- state.raw_data = {
- "ams": [
- {
- "id": 0,
- "module_type": "n3f",
- "dry_time": 0,
- "humidity_raw": "75",
- "dry_sf_reason": [],
- "tray": [{"tray_type": "PLA"}],
- }
- ]
- }
- state.firmware_version = "01.09.00.00"
- mock_pm.get_status.return_value = state
- mock_pm.is_connected.return_value = True
- mock_pm.get_model.return_value = "X1C"
- mock_pm.send_drying_command.return_value = True
- scheduler._is_printer_idle = MagicMock(return_value=True)
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("false"),
- "ambient_drying_enabled": self._make_setting("true"),
- "ams_humidity_fair": self._make_setting("60"),
- "queue_drying_block": self._make_setting("false"),
- "drying_presets": None,
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns))
- # Empty queue — ambient mode should still dry
- await scheduler._check_auto_drying(db, [], set())
- mock_pm.send_drying_command.assert_called_once_with(1, 0, 45, 12, mode=1, filament="PLA")
- assert 1 in scheduler._drying_in_progress
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- @patch("backend.app.services.print_scheduler.supports_drying", return_value=True)
- async def test_ambient_does_not_dry_below_threshold(self, mock_sd, mock_pm, scheduler):
- """Ambient mode does NOT dry when humidity is below threshold."""
- state = MagicMock()
- state.raw_data = {
- "ams": [
- {
- "id": 0,
- "module_type": "n3f",
- "dry_time": 0,
- "humidity_raw": "40",
- "dry_sf_reason": [],
- "tray": [{"tray_type": "PLA"}],
- }
- ]
- }
- state.firmware_version = "01.09.00.00"
- mock_pm.get_status.return_value = state
- mock_pm.is_connected.return_value = True
- mock_pm.get_model.return_value = "X1C"
- scheduler._is_printer_idle = MagicMock(return_value=True)
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("false"),
- "ambient_drying_enabled": self._make_setting("true"),
- "ams_humidity_fair": self._make_setting("60"),
- "queue_drying_block": self._make_setting("false"),
- "drying_presets": None,
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns))
- await scheduler._check_auto_drying(db, [], set())
- mock_pm.send_drying_command.assert_not_called()
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- async def test_ambient_off_stops_drying_without_queue(self, mock_pm, scheduler):
- """Disabling ambient drying stops drying on printers without queue items."""
- scheduler._drying_in_progress = {1: time.monotonic()}
- state = MagicMock()
- state.raw_data = {"ams": [{"id": 0, "dry_time": 120}]}
- mock_pm.get_status.return_value = state
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("false"),
- "ambient_drying_enabled": self._make_setting("false"),
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns))
- await scheduler._check_auto_drying(db, [], set())
- assert mock_pm.send_drying_command.called
- assert not scheduler._drying_in_progress
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- @patch("backend.app.services.print_scheduler.supports_drying", return_value=True)
- async def test_ambient_continues_when_queue_empty(self, mock_sd, mock_pm, scheduler):
- """Ambient drying continues even when queue has no scheduled items (unlike queue mode)."""
- scheduler._drying_in_progress = {1: time.monotonic() - 100}
- state = MagicMock()
- state.raw_data = {
- "ams": [
- {
- "id": 0,
- "module_type": "n3f",
- "dry_time": 600,
- "humidity_raw": "75",
- "dry_sf_reason": [],
- "tray": [{"tray_type": "PLA"}],
- }
- ]
- }
- state.firmware_version = "01.09.00.00"
- mock_pm.get_status.return_value = state
- mock_pm.is_connected.return_value = True
- mock_pm.get_model.return_value = "X1C"
- scheduler._is_printer_idle = MagicMock(return_value=True)
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("false"),
- "ambient_drying_enabled": self._make_setting("true"),
- "ams_humidity_fair": self._make_setting("60"),
- "queue_drying_block": self._make_setting("false"),
- "drying_presets": None,
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns))
- await scheduler._check_auto_drying(db, [], set())
- # Should NOT have sent stop — humidity still high, drying continues
- for call in mock_pm.send_drying_command.call_args_list:
- assert call.kwargs.get("mode") != 0, "Should not stop drying in ambient mode with high humidity"
- assert 1 in scheduler._drying_in_progress
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- @patch("backend.app.services.print_scheduler.supports_drying", return_value=True)
- async def test_queue_only_does_not_dry_without_scheduled_items(self, mock_sd, mock_pm, scheduler):
- """Queue mode alone does NOT dry printers that have no scheduled queue items."""
- state = MagicMock()
- state.raw_data = {
- "ams": [
- {
- "id": 0,
- "module_type": "n3f",
- "dry_time": 0,
- "humidity_raw": "75",
- "dry_sf_reason": [],
- "tray": [{"tray_type": "PLA"}],
- }
- ]
- }
- state.firmware_version = "01.09.00.00"
- mock_pm.get_status.return_value = state
- mock_pm.is_connected.return_value = True
- mock_pm.get_model.return_value = "X1C"
- scheduler._is_printer_idle = MagicMock(return_value=True)
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("true"),
- "ambient_drying_enabled": self._make_setting("false"),
- "ams_humidity_fair": self._make_setting("60"),
- "queue_drying_block": self._make_setting("false"),
- "drying_presets": None,
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns))
- # No queue items at all
- await scheduler._check_auto_drying(db, [], set())
- mock_pm.send_drying_command.assert_not_called()
- class TestBlockForDryingBugFix(_DryingTestBase):
- """Regression: block mode should not skip humidity auto-stop for already-drying printers."""
- @pytest.fixture
- def scheduler(self):
- s = PrintScheduler()
- s._min_drying_seconds = 1800
- return s
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- @patch("backend.app.services.print_scheduler.supports_drying", return_value=True)
- async def test_block_mode_allows_humidity_stop_for_active_drying(self, mock_sd, mock_pm, scheduler):
- """Bug fix: printer already drying in block mode should still check humidity to auto-stop."""
- # Drying started 35 minutes ago
- scheduler._drying_in_progress = {1: time.monotonic() - 2100}
- state = MagicMock()
- state.raw_data = {
- "ams": [
- {
- "id": 0,
- "module_type": "n3f",
- "dry_time": 600,
- "humidity_raw": "30", # Below threshold
- "dry_sf_reason": [],
- "tray": [{"tray_type": "PLA"}],
- }
- ]
- }
- state.firmware_version = "01.09.00.00"
- mock_pm.get_status.return_value = state
- mock_pm.is_connected.return_value = True
- mock_pm.get_model.return_value = "X1C"
- scheduler._is_printer_idle = MagicMock(return_value=True)
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("true"),
- "ambient_drying_enabled": self._make_setting("false"),
- "ams_humidity_fair": self._make_setting("60"),
- "queue_drying_block": self._make_setting("true"),
- "drying_presets": None,
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns))
- # Queue item exists for this printer (triggers block mode gate)
- item = MagicMock()
- item.printer_id = 1
- item.scheduled_time = MagicMock()
- item.manual_start = False
- await scheduler._check_auto_drying(db, [item], set())
- # Should have sent stop command — humidity dropped below threshold after 30+ min
- mock_pm.send_drying_command.assert_any_call(1, 0, temp=0, duration=0, mode=0)
- @pytest.mark.asyncio
- @patch("backend.app.services.print_scheduler.printer_manager")
- @patch("backend.app.services.print_scheduler.supports_drying", return_value=True)
- async def test_block_mode_prevents_new_drying_start(self, mock_sd, mock_pm, scheduler):
- """Block mode should still prevent starting NEW drying on printers with pending items."""
- state = MagicMock()
- state.raw_data = {
- "ams": [
- {
- "id": 0,
- "module_type": "n3f",
- "dry_time": 0,
- "humidity_raw": "75",
- "dry_sf_reason": [],
- "tray": [{"tray_type": "PLA"}],
- }
- ]
- }
- state.firmware_version = "01.09.00.00"
- mock_pm.get_status.return_value = state
- mock_pm.is_connected.return_value = True
- mock_pm.get_model.return_value = "X1C"
- scheduler._is_printer_idle = MagicMock(return_value=True)
- db = AsyncMock()
- settings_returns = {
- "queue_drying_enabled": self._make_setting("true"),
- "ambient_drying_enabled": self._make_setting("false"),
- "ams_humidity_fair": self._make_setting("60"),
- "queue_drying_block": self._make_setting("true"),
- "drying_presets": None,
- }
- db.execute = AsyncMock(side_effect=self._make_db_side_effect(settings_returns))
- item = MagicMock()
- item.printer_id = 1
- item.scheduled_time = MagicMock()
- item.manual_start = False
- await scheduler._check_auto_drying(db, [item], set())
- # Should NOT start drying — block mode with pending items
- mock_pm.send_drying_command.assert_not_called()
|