| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870 |
- """Integration tests for MQTT auto-configuration when assigning a Spoolman spool to an AMS slot.
- Covers:
- - ams_set_filament_setting is called with correct parameters on assign
- - extrusion_cali_sel is called when a matching K-profile exists
- - MQTT failure does NOT roll back the slot assignment
- """
- from unittest.mock import AsyncMock, MagicMock, patch
- import pytest
- from httpx import AsyncClient
- SAMPLE_SPOOL = {
- "id": 10,
- "filament": {
- "id": 1,
- "name": "PLA Basic",
- "material": "PLA",
- "color_hex": "FF0000",
- "weight": 1000,
- "vendor": {"id": 1, "name": "BrandX"},
- },
- "remaining_weight": 800.0,
- "used_weight": 200.0,
- "location": None,
- "comment": None,
- "first_used": None,
- "last_used": None,
- "registered": "2024-01-01T00:00:00+00:00",
- "archived": False,
- "price": None,
- "extra": {},
- }
- @pytest.fixture
- async def slot_settings(db_session):
- from backend.app.models.settings import Settings
- db_session.add(Settings(key="spoolman_enabled", value="true"))
- db_session.add(Settings(key="spoolman_url", value="http://localhost:7912"))
- await db_session.commit()
- @pytest.fixture
- async def test_printer(db_session):
- from backend.app.models.printer import Printer
- printer = Printer(
- name="MQTT Printer",
- serial_number="MQTTTEST001",
- ip_address="192.168.1.200",
- access_code="12345678",
- )
- db_session.add(printer)
- await db_session.commit()
- await db_session.refresh(printer)
- return printer
- @pytest.fixture
- def mock_spoolman_client():
- client = MagicMock()
- client.base_url = "http://localhost:7912"
- client.health_check = AsyncMock(return_value=True)
- client.get_spool = AsyncMock(return_value=SAMPLE_SPOOL)
- # #1457: assign route enumerates spools to clear stale fallback-tag links.
- client.get_spools = AsyncMock(return_value=[])
- client.merge_spool_extra = AsyncMock(return_value={"id": 0, "extra": {}})
- with patch(
- "backend.app.api.routes.spoolman_inventory._get_client",
- AsyncMock(return_value=client),
- ):
- yield client
- class TestAssignSlotMqtt:
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_mqtt_ams_set_filament_called_on_assign(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client
- ):
- """Assigning a Spoolman spool fires ams_set_filament_setting via MQTT."""
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- mqtt_mock.printer_state = None
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- response = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 1,
- },
- )
- assert response.status_code == 200
- mqtt_mock.ams_set_filament_setting.assert_called_once()
- call_kwargs = mqtt_mock.ams_set_filament_setting.call_args[1]
- assert call_kwargs["ams_id"] == 0
- assert call_kwargs["tray_id"] == 1
- assert call_kwargs["tray_type"] == "PLA"
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_mqtt_failure_does_not_rollback_assignment(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client
- ):
- """A crash inside the MQTT block must not un-persist the slot assignment."""
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock(side_effect=RuntimeError("MQTT down"))
- mqtt_mock.printer_state = None
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- response = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 1,
- "tray_id": 0,
- },
- )
- assert response.status_code == 200
- # Verify the assignment IS in the DB despite the MQTT crash
- all_resp = await async_client.get(
- "/api/v1/spoolman/inventory/slot-assignments/all",
- params={"printer_id": test_printer.id},
- )
- assert all_resp.status_code == 200
- rows = all_resp.json()
- assert any(r["spoolman_spool_id"] == 10 for r in rows)
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_extrusion_cali_sel_called_when_k_profile_exists(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client, db_session
- ):
- """extrusion_cali_sel is fired when a matching SpoolmanKProfile row exists."""
- from backend.app.models.spoolman_k_profile import SpoolmanKProfile
- kp = SpoolmanKProfile(
- spoolman_spool_id=10,
- printer_id=test_printer.id,
- extruder=0,
- nozzle_diameter="0.4",
- k_value=0.02,
- cali_idx=5,
- setting_id="CaliID",
- )
- db_session.add(kp)
- await db_session.commit()
- printer_state = MagicMock()
- printer_state.nozzles = [MagicMock(nozzle_diameter="0.4")]
- printer_state.ams_extruder_map = {"0": 0}
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- # Legacy attribute — production never had it set; keep for any code
- # path that still reads `mqtt_client.printer_state` directly. State
- # for the K-profile cascade now comes from printer_manager.get_status.
- mqtt_mock.printer_state = printer_state
- # Empty list = no printer-side kprofiles, so the realignment skips
- # printer_kp lookup. Tests that exercise realignment explicitly
- # populate this list themselves.
- if (
- not hasattr(printer_state, "kprofiles")
- or printer_state.kprofiles is None
- or isinstance(printer_state.kprofiles, MagicMock)
- ):
- printer_state.kprofiles = []
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- response = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 2,
- },
- )
- assert response.status_code == 200
- mqtt_mock.extrusion_cali_sel.assert_called_once()
- call_kwargs = mqtt_mock.extrusion_cali_sel.call_args[1]
- assert call_kwargs["cali_idx"] == 5
- assert call_kwargs["ams_id"] == 0
- assert call_kwargs["tray_id"] == 2
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_extrusion_cali_sel_resets_default_on_nozzle_mismatch(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client, db_session
- ):
- """When nozzle diameter doesn't match K-profile (no usable kp), slot resets to Default K."""
- from backend.app.models.spoolman_k_profile import SpoolmanKProfile
- kp = SpoolmanKProfile(
- spoolman_spool_id=10,
- printer_id=test_printer.id,
- extruder=0,
- nozzle_diameter="0.6",
- k_value=0.03,
- cali_idx=7,
- setting_id="CaliID",
- )
- db_session.add(kp)
- await db_session.commit()
- printer_state = MagicMock()
- printer_state.nozzles = [MagicMock(nozzle_diameter="0.4")]
- printer_state.ams_extruder_map = {"0": 0}
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- # Legacy attribute — production never had it set; keep for any code
- # path that still reads `mqtt_client.printer_state` directly. State
- # for the K-profile cascade now comes from printer_manager.get_status.
- mqtt_mock.printer_state = printer_state
- # Empty list = no printer-side kprofiles, so the realignment skips
- # printer_kp lookup. Tests that exercise realignment explicitly
- # populate this list themselves.
- if (
- not hasattr(printer_state, "kprofiles")
- or printer_state.kprofiles is None
- or isinstance(printer_state.kprofiles, MagicMock)
- ):
- printer_state.kprofiles = []
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- response = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 3,
- },
- )
- assert response.status_code == 200
- mqtt_mock.extrusion_cali_sel.assert_called_once()
- assert mqtt_mock.extrusion_cali_sel.call_args[1]["cali_idx"] == -1
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_extrusion_cali_sel_resets_default_when_cali_idx_none(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client, db_session
- ):
- """When stored K-profile has cali_idx=None (unusable), slot resets to Default K."""
- from backend.app.models.spoolman_k_profile import SpoolmanKProfile
- kp = SpoolmanKProfile(
- spoolman_spool_id=10,
- printer_id=test_printer.id,
- extruder=0,
- nozzle_diameter="0.4",
- k_value=0.02,
- cali_idx=None,
- setting_id=None,
- )
- db_session.add(kp)
- await db_session.commit()
- printer_state = MagicMock()
- printer_state.nozzles = [MagicMock(nozzle_diameter="0.4")]
- printer_state.ams_extruder_map = {"0": 0}
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- # Legacy attribute — production never had it set; keep for any code
- # path that still reads `mqtt_client.printer_state` directly. State
- # for the K-profile cascade now comes from printer_manager.get_status.
- mqtt_mock.printer_state = printer_state
- # Empty list = no printer-side kprofiles, so the realignment skips
- # printer_kp lookup. Tests that exercise realignment explicitly
- # populate this list themselves.
- if (
- not hasattr(printer_state, "kprofiles")
- or printer_state.kprofiles is None
- or isinstance(printer_state.kprofiles, MagicMock)
- ):
- printer_state.kprofiles = []
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- response = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 3,
- },
- )
- assert response.status_code == 200
- mqtt_mock.extrusion_cali_sel.assert_called_once()
- assert mqtt_mock.extrusion_cali_sel.call_args[1]["cali_idx"] == -1
- # ---------------------------------------------------------------------------
- # F7: ams_id=255 External-Slot Extruder-Inversion
- # ---------------------------------------------------------------------------
- class TestExternalSlotExtruderInversion:
- """F7: ams_id=255 maps tray_id→extruder via inversion (0→1, 1→0)."""
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_external_slot_tray0_maps_to_extruder1(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client, db_session
- ):
- """tray_id=0 on ams_id=255 → extruder=1 (ext-L)."""
- from backend.app.models.spoolman_k_profile import SpoolmanKProfile
- # Create K-profiles for both extruders so we can verify which one matches
- kp_extruder_1 = SpoolmanKProfile(
- spoolman_spool_id=10,
- printer_id=test_printer.id,
- extruder=1,
- nozzle_diameter="0.4",
- k_value=0.03,
- cali_idx=1,
- setting_id=None,
- )
- db_session.add(kp_extruder_1)
- await db_session.commit()
- printer_state = MagicMock()
- printer_state.nozzles = [MagicMock(nozzle_diameter="0.4"), MagicMock(nozzle_diameter="0.4")]
- printer_state.ams_extruder_map = {"0": 0} # present so external inversion logic triggers
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- # Legacy attribute — production never had it set; keep for any code
- # path that still reads `mqtt_client.printer_state` directly. State
- # for the K-profile cascade now comes from printer_manager.get_status.
- mqtt_mock.printer_state = printer_state
- # Empty list = no printer-side kprofiles, so the realignment skips
- # printer_kp lookup. Tests that exercise realignment explicitly
- # populate this list themselves.
- if (
- not hasattr(printer_state, "kprofiles")
- or printer_state.kprofiles is None
- or isinstance(printer_state.kprofiles, MagicMock)
- ):
- printer_state.kprofiles = []
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- resp = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 255,
- "tray_id": 0,
- },
- )
- assert resp.status_code == 200
- # extrusion_cali_sel should be called with the K-profile for extruder=1 (cali_idx=1)
- # The extruder itself is not passed as an argument — it's used internally to filter profiles
- mqtt_mock.extrusion_cali_sel.assert_called_once()
- call_kwargs = mqtt_mock.extrusion_cali_sel.call_args[1]
- assert call_kwargs["cali_idx"] == 1
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_external_slot_tray1_maps_to_extruder0(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client, db_session
- ):
- """tray_id=1 on ams_id=255 → extruder=0 (ext-R)."""
- from backend.app.models.spoolman_k_profile import SpoolmanKProfile
- kp_extruder_0 = SpoolmanKProfile(
- spoolman_spool_id=10,
- printer_id=test_printer.id,
- extruder=0,
- nozzle_diameter="0.4",
- k_value=0.02,
- cali_idx=2,
- setting_id=None,
- )
- db_session.add(kp_extruder_0)
- await db_session.commit()
- printer_state = MagicMock()
- printer_state.nozzles = [MagicMock(nozzle_diameter="0.4"), MagicMock(nozzle_diameter="0.4")]
- printer_state.ams_extruder_map = {"0": 0}
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- # Legacy attribute — production never had it set; keep for any code
- # path that still reads `mqtt_client.printer_state` directly. State
- # for the K-profile cascade now comes from printer_manager.get_status.
- mqtt_mock.printer_state = printer_state
- # Empty list = no printer-side kprofiles, so the realignment skips
- # printer_kp lookup. Tests that exercise realignment explicitly
- # populate this list themselves.
- if (
- not hasattr(printer_state, "kprofiles")
- or printer_state.kprofiles is None
- or isinstance(printer_state.kprofiles, MagicMock)
- ):
- printer_state.kprofiles = []
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- resp = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 255,
- "tray_id": 1,
- },
- )
- assert resp.status_code == 200
- # extrusion_cali_sel should be called with the K-profile for extruder=0 (cali_idx=2)
- mqtt_mock.extrusion_cali_sel.assert_called_once()
- call_kwargs = mqtt_mock.extrusion_cali_sel.call_args[1]
- assert call_kwargs["cali_idx"] == 2
- # ---------------------------------------------------------------------------
- # P9-TEST-BE: Live cali_idx fallback when no K-profile is stored (Bug #10)
- # ---------------------------------------------------------------------------
- class TestAssignSpoolmanSlotLiveCaliIdx:
- """When no SpoolmanKProfile exists, live tray cali_idx is used as fallback."""
- def _make_printer_state(self, ams_id: int, tray_id: int, cali_idx: int | None):
- """Build a minimal printer_state mock with one AMS tray."""
- tray_mock = {
- "id": tray_id,
- "cali_idx": cali_idx,
- }
- ams_mock = {"id": ams_id, "tray": [tray_mock]}
- state = MagicMock()
- state.nozzles = [MagicMock(nozzle_diameter="0.4")]
- state.ams_extruder_map = {str(ams_id): 0}
- state.raw_data = {"ams": [ams_mock]}
- return state
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_no_kprofile_resets_to_default_k(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client
- ):
- """When no K-profile exists, slot resets to cali_idx=-1 (Default K) regardless of live value."""
- printer_state = self._make_printer_state(ams_id=0, tray_id=1, cali_idx=42)
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- # Legacy attribute — production never had it set; keep for any code
- # path that still reads `mqtt_client.printer_state` directly. State
- # for the K-profile cascade now comes from printer_manager.get_status.
- mqtt_mock.printer_state = printer_state
- # Empty list = no printer-side kprofiles, so the realignment skips
- # printer_kp lookup. Tests that exercise realignment explicitly
- # populate this list themselves.
- if (
- not hasattr(printer_state, "kprofiles")
- or printer_state.kprofiles is None
- or isinstance(printer_state.kprofiles, MagicMock)
- ):
- printer_state.kprofiles = []
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- resp = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 1,
- },
- )
- assert resp.status_code == 200
- mqtt_mock.extrusion_cali_sel.assert_called_once()
- call_kwargs = mqtt_mock.extrusion_cali_sel.call_args[1]
- assert call_kwargs["cali_idx"] == -1
- assert call_kwargs["ams_id"] == 0
- assert call_kwargs["tray_id"] == 1
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_no_kprofile_no_live_cali_idx_sends_default(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client
- ):
- """When no K-profile and tray has no cali_idx, extrusion_cali_sel is sent with cali_idx=-1 (Default)."""
- printer_state = self._make_printer_state(ams_id=0, tray_id=2, cali_idx=None)
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- # Legacy attribute — production never had it set; keep for any code
- # path that still reads `mqtt_client.printer_state` directly. State
- # for the K-profile cascade now comes from printer_manager.get_status.
- mqtt_mock.printer_state = printer_state
- # Empty list = no printer-side kprofiles, so the realignment skips
- # printer_kp lookup. Tests that exercise realignment explicitly
- # populate this list themselves.
- if (
- not hasattr(printer_state, "kprofiles")
- or printer_state.kprofiles is None
- or isinstance(printer_state.kprofiles, MagicMock)
- ):
- printer_state.kprofiles = []
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- resp = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 2,
- },
- )
- assert resp.status_code == 200
- mqtt_mock.extrusion_cali_sel.assert_called_once()
- assert mqtt_mock.extrusion_cali_sel.call_args[1]["cali_idx"] == -1
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_kprofile_takes_priority_over_live_cali_idx(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client, db_session
- ):
- """Stored K-profile cali_idx wins over live tray cali_idx."""
- from backend.app.models.spoolman_k_profile import SpoolmanKProfile
- kp = SpoolmanKProfile(
- spoolman_spool_id=10,
- printer_id=test_printer.id,
- extruder=0,
- nozzle_diameter="0.4",
- k_value=0.02,
- cali_idx=10,
- setting_id="CaliID",
- )
- db_session.add(kp)
- await db_session.commit()
- # Live tray has a different cali_idx — stored profile must win
- printer_state = self._make_printer_state(ams_id=0, tray_id=3, cali_idx=99)
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- # Legacy attribute — production never had it set; keep for any code
- # path that still reads `mqtt_client.printer_state` directly. State
- # for the K-profile cascade now comes from printer_manager.get_status.
- mqtt_mock.printer_state = printer_state
- # Empty list = no printer-side kprofiles, so the realignment skips
- # printer_kp lookup. Tests that exercise realignment explicitly
- # populate this list themselves.
- if (
- not hasattr(printer_state, "kprofiles")
- or printer_state.kprofiles is None
- or isinstance(printer_state.kprofiles, MagicMock)
- ):
- printer_state.kprofiles = []
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- resp = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 3,
- },
- )
- assert resp.status_code == 200
- mqtt_mock.extrusion_cali_sel.assert_called_once()
- call_kwargs = mqtt_mock.extrusion_cali_sel.call_args[1]
- # Must use stored K-profile (10), NOT live cali_idx (99)
- assert call_kwargs["cali_idx"] == 10
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_live_cali_idx_negative_falls_back_to_default(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client
- ):
- """A negative live cali_idx falls through and is sent as Default (cali_idx=-1)."""
- printer_state = self._make_printer_state(ams_id=0, tray_id=0, cali_idx=-1)
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- # Legacy attribute — production never had it set; keep for any code
- # path that still reads `mqtt_client.printer_state` directly. State
- # for the K-profile cascade now comes from printer_manager.get_status.
- mqtt_mock.printer_state = printer_state
- # Empty list = no printer-side kprofiles, so the realignment skips
- # printer_kp lookup. Tests that exercise realignment explicitly
- # populate this list themselves.
- if (
- not hasattr(printer_state, "kprofiles")
- or printer_state.kprofiles is None
- or isinstance(printer_state.kprofiles, MagicMock)
- ):
- printer_state.kprofiles = []
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- resp = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 0,
- },
- )
- assert resp.status_code == 200
- mqtt_mock.extrusion_cali_sel.assert_called_once()
- assert mqtt_mock.extrusion_cali_sel.call_args[1]["cali_idx"] == -1
- # ---------------------------------------------------------------------------
- # Realignment of slot filament context to K-profile preset
- # ---------------------------------------------------------------------------
- # When the user assigns a Spoolman spool whose stored kp was calibrated under
- # a specific filament preset (e.g. P-prefix local, or a named cloud preset),
- # the slot must be configured under THAT preset for the printer to find the
- # cali_idx in its calibration table. Without realignment the slot ends up on
- # generic PLA / default K — the symptom maztiggy reported on x1c-2 (#1114).
- class TestAssignSpoolmanSlotKProfileRealignment:
- """assign_spoolman_slot realigns tray_info_idx + setting_id to kp context."""
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_realigns_to_printer_reported_filament_id(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client, db_session
- ):
- """When state.kprofiles has the cali_idx, use printer_kp.filament_id verbatim.
- The printer keys its calibration table by filament_id, not setting_id.
- For a P-prefix local preset (printer-registered), filament_id and
- tray_info_idx must match for the cali_idx to apply.
- """
- from backend.app.models.spoolman_k_profile import SpoolmanKProfile
- # Stored kp with setting_id but no filament_id (the schema gap)
- kp = SpoolmanKProfile(
- spoolman_spool_id=10,
- printer_id=test_printer.id,
- extruder=0,
- nozzle_diameter="0.4",
- k_value=0.025,
- cali_idx=8948,
- setting_id="PFUSedbf16b803ff3e",
- )
- db_session.add(kp)
- await db_session.commit()
- printer_state = MagicMock()
- printer_state.nozzles = [MagicMock(nozzle_diameter="0.4")]
- printer_state.ams_extruder_map = {"0": 0}
- printer_state.raw_data = None
- # Live calibration entry from the printer — this is what cali_idx 8948
- # is actually registered under. P-prefix is a printer-local preset
- # (different from PFUS-prefix cloud user presets).
- printer_kp = MagicMock()
- printer_kp.slot_id = 8948
- printer_kp.nozzle_diameter = "0.4"
- printer_kp.filament_id = "P4d64437"
- printer_kp.setting_id = "PFUSedbf16b803ff3e"
- printer_state.kprofiles = [printer_kp]
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- mqtt_mock.printer_state = printer_state
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- response = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 1,
- },
- )
- assert response.status_code == 200
- # Both MQTT commands must reference the printer-reported filament_id
- # so the slot context and the cali_sel context match.
- amf_kwargs = mqtt_mock.ams_set_filament_setting.call_args[1]
- assert amf_kwargs["tray_info_idx"] == "P4d64437"
- assert amf_kwargs["setting_id"] == "PFUSedbf16b803ff3e"
- cs_kwargs = mqtt_mock.extrusion_cali_sel.call_args[1]
- assert cs_kwargs["cali_idx"] == 8948
- assert cs_kwargs["filament_id"] == "P4d64437"
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_skips_realignment_for_pfus_prefix(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client, db_session
- ):
- """PFUS-prefix cloud-user presets are rejected by the slicer in tray_info_idx.
- For those, tray_info_idx must stay as the GF* generic so the slicer
- can render the slot. setting_id can still be realigned to the cloud
- preset (slicer uses that for display), but tray_info_idx stays GF*.
- """
- from backend.app.models.spoolman_k_profile import SpoolmanKProfile
- kp = SpoolmanKProfile(
- spoolman_spool_id=10,
- printer_id=test_printer.id,
- extruder=0,
- nozzle_diameter="0.4",
- k_value=0.025,
- cali_idx=42,
- setting_id="PFUSedbf16b803ff3e",
- )
- db_session.add(kp)
- await db_session.commit()
- printer_state = MagicMock()
- printer_state.nozzles = [MagicMock(nozzle_diameter="0.4")]
- printer_state.ams_extruder_map = {"0": 0}
- printer_state.raw_data = None
- # Printer-side kp filament_id is PFUS-prefix → realignment must skip
- printer_kp = MagicMock()
- printer_kp.slot_id = 42
- printer_kp.nozzle_diameter = "0.4"
- printer_kp.filament_id = "PFUSedbf16b803ff3e"
- printer_kp.setting_id = "PFUSedbf16b803ff3e"
- printer_state.kprofiles = [printer_kp]
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- mqtt_mock.printer_state = printer_state
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- response = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 2,
- },
- )
- assert response.status_code == 200
- amf_kwargs = mqtt_mock.ams_set_filament_setting.call_args[1]
- # tray_info_idx stays as the resolved generic (slicer accepts GF*)
- assert amf_kwargs["tray_info_idx"] == "GFL99"
- # setting_id may be realigned to the cloud preset for slicer display
- assert amf_kwargs["setting_id"] == "PFUSedbf16b803ff3e"
- @pytest.mark.asyncio
- @pytest.mark.integration
- async def test_extruder_relax_falls_back_to_any_extruder_kp(
- self, async_client: AsyncClient, slot_settings, test_printer, mock_spoolman_client, db_session
- ):
- """Hard-skip on extruder mismatch silently dropped valid stored profiles
- when the AMS-extruder map shifted. The cascade now prefers exact
- extruder match but falls back to any kp on the same printer + nozzle.
- """
- from backend.app.models.spoolman_k_profile import SpoolmanKProfile
- # kp is for extruder=1, but slot will be on extruder=0 (mismatch)
- kp = SpoolmanKProfile(
- spoolman_spool_id=10,
- printer_id=test_printer.id,
- extruder=1,
- nozzle_diameter="0.4",
- k_value=0.025,
- cali_idx=42,
- setting_id="GFSL05",
- )
- db_session.add(kp)
- await db_session.commit()
- printer_state = MagicMock()
- printer_state.nozzles = [MagicMock(nozzle_diameter="0.4")]
- printer_state.ams_extruder_map = {"0": 0}
- printer_state.raw_data = None
- printer_state.kprofiles = []
- mqtt_mock = MagicMock()
- mqtt_mock.ams_set_filament_setting = MagicMock()
- mqtt_mock.extrusion_cali_sel = MagicMock()
- mqtt_mock.printer_state = printer_state
- with patch("backend.app.api.routes.spoolman_inventory.printer_manager") as pm_mock:
- pm_mock.get_client = MagicMock(return_value=mqtt_mock)
- pm_mock.get_status = MagicMock(return_value=printer_state)
- response = await async_client.post(
- "/api/v1/spoolman/inventory/slot-assignments",
- json={
- "spoolman_spool_id": 10,
- "printer_id": test_printer.id,
- "ams_id": 0,
- "tray_id": 3,
- },
- )
- assert response.status_code == 200
- # extruder mismatch was hard-skipped pre-fix; now used as fallback
- cs_kwargs = mqtt_mock.extrusion_cali_sel.call_args[1]
- assert cs_kwargs["cali_idx"] == 42
|