| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667 |
- """Unit tests for Virtual Printer services.
- Tests the virtual printer manager, FTP server, and SSDP server components.
- """
- import asyncio
- import json
- import zipfile
- from pathlib import Path
- from unittest.mock import AsyncMock, MagicMock, patch
- import pytest
- def _write_3mf_with_filaments(file_path: Path, filaments: list[dict], plate_index: int = 1) -> None:
- """Build a minimal 3MF zip with `Metadata/slice_info.config` carrying the
- given per-slot filament entries. Each `filaments` dict needs `id`, `type`,
- `color`, `used_g`. Used by the #1188 VP queue-mode tests below."""
- filament_xml = "".join(
- f'<filament id="{f["id"]}" type="{f["type"]}" color="{f["color"]}" '
- f'used_g="{f["used_g"]}" tray_info_idx="{f.get("tray_info_idx", "")}"/>'
- for f in filaments
- )
- config = (
- '<?xml version="1.0" encoding="utf-8"?>'
- "<config>"
- f'<plate><metadata key="index" value="{plate_index}"/>'
- f"{filament_xml}"
- "</plate>"
- "</config>"
- )
- with zipfile.ZipFile(file_path, "w") as zf:
- zf.writestr("Metadata/slice_info.config", config)
- # Plate gcode is referenced for plate-id detection in the VP path —
- # presence is enough; contents don't matter.
- zf.writestr(f"Metadata/plate_{plate_index}.gcode", "; gcode\n")
- class TestVirtualPrinterInstance:
- """Tests for VirtualPrinterInstance class."""
- @pytest.fixture
- def instance(self, tmp_path):
- """Create a VirtualPrinterInstance with test defaults."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- return VirtualPrinterInstance(
- vp_id=1,
- name="TestPrinter",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- base_dir=tmp_path,
- )
- # ========================================================================
- # Tests for instance properties
- # ========================================================================
- def test_instance_stores_parameters(self, instance):
- """Verify constructor stores parameters correctly."""
- assert instance.id == 1
- assert instance.name == "TestPrinter"
- assert instance.mode == "immediate"
- assert instance.model == "C11"
- assert instance.access_code == "12345678"
- assert instance.serial_suffix == "391800001"
- def test_instance_serial_property(self, instance):
- """Verify serial is generated from model prefix + suffix."""
- # C11 = P1P, prefix = 01S00A
- assert instance.serial == "01S00A391800001"
- def test_instance_serial_x1c(self, tmp_path):
- """Verify X1C serial uses correct prefix."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=2,
- name="X1C",
- mode="immediate",
- model="BL-P001",
- access_code="12345678",
- serial_suffix="391800002",
- base_dir=tmp_path,
- )
- assert inst.serial == "00M00A391800002"
- def test_instance_is_proxy_false(self, instance):
- """Verify is_proxy is False for non-proxy mode."""
- assert instance.is_proxy is False
- def test_instance_is_proxy_true(self, tmp_path):
- """Verify is_proxy is True for proxy mode."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=3,
- name="Proxy",
- mode="proxy",
- model="C11",
- access_code="",
- serial_suffix="391800003",
- target_printer_ip="192.168.1.100",
- base_dir=tmp_path,
- )
- assert inst.is_proxy is True
- def test_instance_is_running_with_active_tasks(self, instance):
- """Verify is_running is True when tasks are active."""
- mock_task = MagicMock()
- mock_task.done.return_value = False
- instance._tasks = [mock_task]
- assert instance.is_running is True
- def test_instance_is_running_with_no_tasks(self, instance):
- """Verify is_running is False when no tasks."""
- assert instance.is_running is False
- def test_instance_creates_directories(self, instance, tmp_path):
- """Verify instance creates upload and cert directories."""
- assert (tmp_path / "uploads" / "1").exists()
- assert (tmp_path / "uploads" / "1" / "cache").exists()
- assert (tmp_path / "certs" / "1").exists()
- # ========================================================================
- # Tests for status
- # ========================================================================
- def test_get_status_returns_correct_format(self, instance):
- """Verify get_status returns expected fields."""
- instance._pending_files = {"file1.3mf": Path("/tmp/file1.3mf")} # nosec B108
- mock_task = MagicMock(done=MagicMock(return_value=False))
- instance._tasks = [mock_task]
- status = instance.get_status()
- assert status["running"] is True
- assert status["pending_files"] == 1
- def test_get_status_not_running(self, instance):
- """Verify get_status when no tasks."""
- status = instance.get_status()
- assert status["running"] is False
- assert status["pending_files"] == 0
- # ========================================================================
- # Tests for file handling
- # ========================================================================
- @pytest.mark.asyncio
- async def test_on_file_received_adds_to_pending(self, instance):
- """Verify received file is added to pending list in review mode."""
- instance.mode = "review"
- file_path = Path("/tmp/test.3mf") # nosec B108
- with patch.object(instance, "_queue_file", new_callable=AsyncMock) as mock_queue:
- await instance.on_file_received(file_path, "192.168.1.100")
- assert "test.3mf" in instance._pending_files
- mock_queue.assert_called_once()
- @pytest.mark.asyncio
- async def test_on_file_received_archives_immediately(self, instance):
- """Verify file is archived in immediate mode."""
- file_path = Path("/tmp/test.3mf") # nosec B108
- with patch.object(instance, "_archive_file", new_callable=AsyncMock) as mock_archive:
- await instance.on_file_received(file_path, "192.168.1.100")
- mock_archive.assert_called_once_with(file_path, "192.168.1.100")
- @pytest.mark.asyncio
- async def test_on_file_received_signals_FINISH_to_slicer(self, instance):
- """Regression #1280: when a slicer's Print flow uploads to a non-proxy VP,
- the VP must transition gcode_state PREPARE → FINISH so the slicer's
- in-flight-job lock releases. Going PREPARE → IDLE wedges Orca at
- "Downloading...(0%)" and blocks the next dispatch with "busy with
- another print job".
- Send-flow slicers don't watch the post-upload state, so this is a
- no-op behavior change for them.
- """
- instance.mode = "immediate"
- instance._mqtt = MagicMock()
- instance._mqtt.set_gcode_state = MagicMock()
- file_path = Path("/tmp/test.3mf") # nosec B108
- with patch.object(instance, "_archive_file", new_callable=AsyncMock):
- await instance.on_file_received(file_path, "192.168.1.100")
- instance._mqtt.set_gcode_state.assert_called_once_with("FINISH", filename="test.3mf", prepare_percent="100")
- @pytest.mark.asyncio
- async def test_on_file_received_non_3mf_does_not_touch_state(self, instance):
- """Non-3MF uploads (e.g., a job's auxiliary files) must not transition
- the visible state — the slicer is only tracking the .3mf upload."""
- instance.mode = "immediate"
- instance._mqtt = MagicMock()
- instance._mqtt.set_gcode_state = MagicMock()
- file_path = Path("/tmp/test.gcode") # nosec B108
- with patch.object(instance, "_archive_file", new_callable=AsyncMock):
- await instance.on_file_received(file_path, "192.168.1.100")
- instance._mqtt.set_gcode_state.assert_not_called()
- @pytest.mark.asyncio
- async def test_archive_file_skips_non_3mf(self, instance):
- """Verify non-3MF files are skipped and cleaned up."""
- instance._session_factory = MagicMock()
- instance._pending_files["verify_job"] = Path("/tmp/verify_job") # nosec B108
- with patch("pathlib.Path.unlink"):
- await instance._archive_file(Path("/tmp/verify_job"), "192.168.1.100") # nosec B108
- assert "verify_job" not in instance._pending_files
- @pytest.mark.asyncio
- async def test_archive_file_broadcasts_archive_created(self, tmp_path):
- """#1282: VP immediate-mode archives must broadcast archive_created so
- the Archives page refreshes without a tab switch. Real-printer prints
- get this via main.py's MQTT print_start handler; the VP path used to
- skip the broadcast entirely."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- mock_db = AsyncMock()
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=30,
- name="ImmediateBroadcast",
- mode="immediate",
- model="C12",
- access_code="12345678",
- serial_suffix="391800030",
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "test.3mf"
- file_path.write_bytes(b"fake3mf")
- mock_archive = MagicMock()
- mock_archive.id = 99
- mock_archive.printer_id = None
- mock_archive.filename = "test.3mf"
- mock_archive.print_name = "test"
- mock_archive.status = "archived"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=None,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- patch(
- "backend.app.core.websocket.ws_manager.send_archive_created",
- new_callable=AsyncMock,
- ) as mock_broadcast,
- ):
- await inst._archive_file(file_path, "192.168.1.100")
- mock_broadcast.assert_awaited_once()
- payload = mock_broadcast.await_args.args[0]
- assert payload["id"] == 99
- assert payload["filename"] == "test.3mf"
- assert payload["status"] == "archived"
- # ========================================================================
- # Tests for auto_dispatch
- # ========================================================================
- def test_auto_dispatch_defaults_to_true(self, tmp_path):
- """Verify auto_dispatch defaults to True when not specified."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=10,
- name="DefaultDispatch",
- mode="print_queue",
- model="C11",
- access_code="12345678",
- serial_suffix="391800010",
- base_dir=tmp_path,
- )
- assert inst.auto_dispatch is True
- @pytest.mark.asyncio
- async def test_add_to_print_queue_with_auto_dispatch_on(self, tmp_path):
- """Verify queue items have manual_start=False when auto_dispatch=True."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- mock_db = AsyncMock()
- added_items = []
- def capture_add(item):
- added_items.append(item)
- mock_db.add = MagicMock(side_effect=capture_add)
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=11,
- name="AutoDispatchOn",
- mode="print_queue",
- model="C11",
- access_code="12345678",
- serial_suffix="391800011",
- auto_dispatch=True,
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- # Create a temp 3mf file
- file_path = tmp_path / "test.3mf"
- file_path.write_bytes(b"fake3mf")
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "test"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=None,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- assert len(added_items) == 1
- queue_item = added_items[0]
- assert queue_item.manual_start is False
- @pytest.mark.asyncio
- async def test_add_to_print_queue_broadcasts_archive_created(self, tmp_path):
- """#1282: VP queue-mode uploads must broadcast archive_created so the
- Archives page picks up the new entry live. Pre-fix the page only
- refreshed when the user manually switched tabs."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- mock_db = AsyncMock()
- mock_db.add = MagicMock()
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=31,
- name="QueueBroadcast",
- mode="print_queue",
- model="C12",
- access_code="12345678",
- serial_suffix="391800031",
- auto_dispatch=True,
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "test.3mf"
- file_path.write_bytes(b"fake3mf")
- mock_archive = MagicMock()
- mock_archive.id = 77
- mock_archive.printer_id = None
- mock_archive.filename = "test.3mf"
- mock_archive.print_name = "test"
- mock_archive.status = "archived"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=None,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- patch(
- "backend.app.core.websocket.ws_manager.send_archive_created",
- new_callable=AsyncMock,
- ) as mock_broadcast,
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- mock_broadcast.assert_awaited_once()
- payload = mock_broadcast.await_args.args[0]
- assert payload["id"] == 77
- assert payload["print_name"] == "test"
- assert payload["status"] == "archived"
- @pytest.mark.asyncio
- async def test_add_to_print_queue_with_auto_dispatch_off(self, tmp_path):
- """Verify queue items have manual_start=True when auto_dispatch=False."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- mock_db = AsyncMock()
- added_items = []
- def capture_add(item):
- added_items.append(item)
- mock_db.add = MagicMock(side_effect=capture_add)
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=12,
- name="AutoDispatchOff",
- mode="print_queue",
- model="C11",
- access_code="12345678",
- serial_suffix="391800012",
- auto_dispatch=False,
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- # Create a temp 3mf file
- file_path = tmp_path / "test.3mf"
- file_path.write_bytes(b"fake3mf")
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "test"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=None,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- assert len(added_items) == 1
- queue_item = added_items[0]
- assert queue_item.manual_start is True
- @pytest.mark.asyncio
- async def test_add_to_print_queue_uses_workflow_defaults_from_settings(self, tmp_path):
- """#1235: VP queue-mode constructed PrintQueueItem without specifying
- bed_levelling / flow_cali / vibration_cali / layer_inspect / timelapse,
- so SQLAlchemy applied the column-level defaults and ignored the user's
- workflow preferences entirely. Every print sent from the slicer to the
- VP came through with the OPPOSITE of what the workflow page said,
- forcing the user to edit each queue item by hand.
- """
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- added_items = []
- mock_db = AsyncMock()
- mock_db.add = MagicMock(side_effect=added_items.append)
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=22,
- name="DefaultsTest",
- mode="print_queue",
- model="C12",
- access_code="12345678",
- serial_suffix="391800022",
- auto_dispatch=True,
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "test.3mf"
- file_path.write_bytes(b"fake3mf")
- # The reporter set every workflow default to the OPPOSITE of the model's
- # column default. Pre-fix the column defaults won; with the fix the
- # settings values must flow through to the queue item exactly as stored.
- settings_map = {
- "virtual_printer_archive_name_source": None,
- "default_bed_levelling": "false", # model default: True
- "default_flow_cali": "true", # model default: False
- "default_vibration_cali": "false", # model default: True
- "default_layer_inspect": "true", # model default: False
- "default_timelapse": "true", # model default: False
- }
- async def fake_get_setting(_db, key):
- return settings_map.get(key)
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "test"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new=fake_get_setting,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- assert len(added_items) == 1
- queue_item = added_items[0]
- assert queue_item.bed_levelling is False, "default_bed_levelling=false must flow through"
- assert queue_item.flow_cali is True, "default_flow_cali=true must flow through"
- assert queue_item.vibration_cali is False, "default_vibration_cali=false must flow through"
- assert queue_item.layer_inspect is True, "default_layer_inspect=true must flow through"
- assert queue_item.timelapse is True, "default_timelapse=true must flow through"
- @pytest.mark.asyncio
- async def test_add_to_print_queue_falls_back_to_schema_defaults_when_unset(self, tmp_path):
- """#1235 fallback: when no workflow setting is in the DB, the queue
- item should use the AppSettings (Pydantic) defaults — same values
- the user sees in the workflow page on a fresh install.
- """
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- added_items = []
- mock_db = AsyncMock()
- mock_db.add = MagicMock(side_effect=added_items.append)
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=23,
- name="FreshInstallDefaults",
- mode="print_queue",
- model="C12",
- access_code="12345678",
- serial_suffix="391800023",
- auto_dispatch=True,
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "test.3mf"
- file_path.write_bytes(b"fake3mf")
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "test"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=None, # No settings → fall back to schema defaults
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- assert len(added_items) == 1
- queue_item = added_items[0]
- # These must match the AppSettings (Pydantic) defaults in schemas/settings.py
- assert queue_item.bed_levelling is True
- assert queue_item.flow_cali is False
- assert queue_item.vibration_cali is True
- assert queue_item.layer_inspect is False
- assert queue_item.timelapse is False
- @pytest.mark.asyncio
- async def test_add_to_print_queue_inherits_slicer_print_options(self, tmp_path):
- """#1403: VP-queue items used to fall back to `default_timelapse` even
- though the slicer's MQTT `project_file` command carries the user's
- actual choice. Capture-via-`on_print_command` flow lets the user's
- slicer toggle reach the queue item.
- Settings here have timelapse OFF; the slicer's MQTT capture has it ON.
- After the fix the queue item must reflect the slicer's choice.
- """
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- added_items = []
- mock_db = AsyncMock()
- mock_db.add = MagicMock(side_effect=added_items.append)
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=24,
- name="SlicerInherits",
- mode="print_queue",
- model="C12",
- access_code="12345678",
- serial_suffix="391800024",
- auto_dispatch=True,
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "test.3mf"
- file_path.write_bytes(b"fake3mf")
- # Pre-populate the capture as if MQTT `project_file` arrived already.
- # Settings (below) deliberately have timelapse OFF — only the slicer
- # capture should drive the resulting queue item.
- await inst.on_print_command(
- file_path.name,
- {
- "command": "project_file",
- "timelapse": True,
- "bed_leveling": False, # Note: MQTT field is single-L `bed_leveling`
- "flow_cali": True,
- "vibration_cali": False,
- "layer_inspect": True,
- },
- )
- settings_map = {
- "virtual_printer_archive_name_source": None,
- "default_bed_levelling": "true",
- "default_flow_cali": "false",
- "default_vibration_cali": "true",
- "default_layer_inspect": "false",
- "default_timelapse": "false",
- }
- async def fake_get_setting(_db, key):
- return settings_map.get(key)
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "test"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new=fake_get_setting,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- assert len(added_items) == 1
- queue_item = added_items[0]
- assert queue_item.timelapse is True, "Slicer's timelapse=True must override settings.default_timelapse=False"
- assert queue_item.bed_levelling is False, "Slicer's bed_leveling=False must override default_bed_levelling=True"
- assert queue_item.flow_cali is True
- assert queue_item.vibration_cali is False
- assert queue_item.layer_inspect is True
- # Capture is consumed — no lingering state for the next print of the same name.
- assert file_path.name not in inst._slicer_print_options
- @pytest.mark.asyncio
- async def test_add_to_print_queue_coerces_slicer_integer_zero_one(self, tmp_path):
- """#1403: H-family firmwares carry calibration flags as integers
- (0/1) rather than booleans. The capture must coerce both shapes so
- H-family-sliced jobs through the VP queue work the same as P1/X1.
- """
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- added_items = []
- mock_db = AsyncMock()
- mock_db.add = MagicMock(side_effect=added_items.append)
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=25,
- name="SlicerIntegers",
- mode="print_queue",
- model="C12",
- access_code="12345678",
- serial_suffix="391800025",
- auto_dispatch=True,
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "test.3mf"
- file_path.write_bytes(b"fake3mf")
- await inst.on_print_command(
- file_path.name,
- {"command": "project_file", "timelapse": 1, "bed_leveling": 0, "flow_cali": 1},
- )
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "test"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=None,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- assert len(added_items) == 1
- queue_item = added_items[0]
- assert queue_item.timelapse is True, "integer 1 must coerce to True"
- assert queue_item.bed_levelling is False, "integer 0 must coerce to False"
- assert queue_item.flow_cali is True
- @pytest.mark.asyncio
- async def test_add_to_print_queue_populates_required_filament_types(self, tmp_path):
- """#1188: VP queue-mode used to create PrintQueueItems with no
- filament fields, so the scheduler fell through to model-only matching
- and dispatched onto whatever printer was free regardless of loaded
- colour. ``required_filament_types`` is populated unconditionally
- (cheap, helps the scheduler validate type even without
- ``force_color_match``) — pin that contract here."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- added_items = []
- mock_db = AsyncMock()
- mock_db.add = MagicMock(side_effect=added_items.append)
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=21,
- name="Reqs",
- mode="print_queue",
- model="C12",
- access_code="12345678",
- serial_suffix="391800021",
- auto_dispatch=True,
- queue_force_color_match=False, # off → only required_filament_types
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "multi.3mf"
- _write_3mf_with_filaments(
- file_path,
- [
- {"id": "1", "type": "PLA", "color": "#FFFFFF", "used_g": "12.3"},
- {"id": "2", "type": "PETG", "color": "#000000", "used_g": "4.5"},
- # used_g=0 → not actually consumed by this plate, must be ignored
- {"id": "3", "type": "ABS", "color": "#FF0000", "used_g": "0"},
- ],
- plate_index=1,
- )
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "multi"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=None,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- assert len(added_items) == 1
- queue_item = added_items[0]
- # Type-only fallback always populated. Sorted, deduped, no zero-use ABS.
- assert queue_item.required_filament_types is not None
- assert json.loads(queue_item.required_filament_types) == ["PETG", "PLA"]
- # Setting off → no force_color_match overrides leaked.
- assert queue_item.filament_overrides is None
- @pytest.mark.asyncio
- async def test_add_to_print_queue_force_color_match_writes_overrides(self, tmp_path):
- """#1188 core fix: when the per-VP ``queue_force_color_match`` toggle
- is on, every consumed slot lands as a ``filament_overrides`` entry
- with ``force_color_match: true``. This is the field the scheduler
- keys on (``print_scheduler.py:512``) — without it, slot-by-slot
- type+color matching never runs."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- added_items = []
- mock_db = AsyncMock()
- mock_db.add = MagicMock(side_effect=added_items.append)
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=22,
- name="ForceColor",
- mode="print_queue",
- model="C12",
- access_code="12345678",
- serial_suffix="391800022",
- auto_dispatch=True,
- queue_force_color_match=True, # on
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "forced.3mf"
- _write_3mf_with_filaments(
- file_path,
- [
- {"id": "1", "type": "PLA", "color": "#FFFFFF", "used_g": "10.0"},
- {"id": "2", "type": "PLA", "color": "#FF00FF", "used_g": "5.0"},
- ],
- plate_index=1,
- )
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "forced"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=None,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- assert len(added_items) == 1
- queue_item = added_items[0]
- assert queue_item.filament_overrides is not None
- overrides = json.loads(queue_item.filament_overrides)
- assert overrides == [
- {"slot_id": 1, "type": "PLA", "color": "#FFFFFF", "force_color_match": True},
- {"slot_id": 2, "type": "PLA", "color": "#FF00FF", "force_color_match": True},
- ]
- # required_filament_types still populated alongside overrides.
- assert json.loads(queue_item.required_filament_types) == ["PLA"]
- @pytest.mark.asyncio
- async def test_add_to_print_queue_force_color_match_skips_when_3mf_unparseable(self, tmp_path):
- """A malformed or fake-bytes 3MF must not crash the upload path —
- we just write the queue item with no filament fields and let the
- scheduler fall back to model-only matching (the pre-#1188 default).
- Regression guard for the existing fake-bytes happy-path tests."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- added_items = []
- mock_db = AsyncMock()
- mock_db.add = MagicMock(side_effect=added_items.append)
- mock_db.commit = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=23,
- name="Unparseable",
- mode="print_queue",
- model="C12",
- access_code="12345678",
- serial_suffix="391800023",
- auto_dispatch=True,
- queue_force_color_match=True,
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "bad.3mf"
- file_path.write_bytes(b"not a real 3mf zip")
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "bad"
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=None,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- new_callable=AsyncMock,
- return_value=mock_archive,
- ),
- ):
- await inst._add_to_print_queue(file_path, "192.168.1.100")
- assert len(added_items) == 1
- queue_item = added_items[0]
- # No filament data extractable → both fields stay None (graceful
- # fallback to model-only scheduling).
- assert queue_item.required_filament_types is None
- assert queue_item.filament_overrides is None
- # ========================================================================
- # Tests for archive_name_source setting (#1152)
- # ========================================================================
- @pytest.mark.asyncio
- @pytest.mark.parametrize(
- ("setting_value", "expected_prefer_filename"),
- [
- ("filename", True),
- ("metadata", False),
- (None, False), # Default when setting unset
- ("", False), # Defensive: empty string is not "filename"
- ],
- )
- async def test_archive_file_passes_prefer_filename_per_setting(
- self, tmp_path, setting_value, expected_prefer_filename
- ):
- """_archive_file reads `virtual_printer_archive_name_source` and forwards
- prefer_filename_for_name=True only when it equals 'filename' (#1152)."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- mock_db = AsyncMock()
- mock_session_factory = MagicMock()
- mock_session_ctx = AsyncMock()
- mock_session_ctx.__aenter__ = AsyncMock(return_value=mock_db)
- mock_session_ctx.__aexit__ = AsyncMock(return_value=False)
- mock_session_factory.return_value = mock_session_ctx
- inst = VirtualPrinterInstance(
- vp_id=20,
- name="NameSource",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800020",
- base_dir=tmp_path,
- session_factory=mock_session_factory,
- )
- file_path = tmp_path / "user-renamed-job.3mf"
- file_path.write_bytes(b"fake3mf")
- mock_archive = MagicMock()
- mock_archive.id = 1
- mock_archive.print_name = "user-renamed-job"
- archive_print_mock = AsyncMock(return_value=mock_archive)
- with (
- patch(
- "backend.app.api.routes.settings.get_setting",
- new_callable=AsyncMock,
- return_value=setting_value,
- ),
- patch(
- "backend.app.services.archive.ArchiveService.archive_print",
- archive_print_mock,
- ),
- ):
- await inst._archive_file(file_path, "192.168.1.100")
- assert archive_print_mock.await_count == 1
- kwargs = archive_print_mock.await_args.kwargs
- assert kwargs.get("prefer_filename_for_name") is expected_prefer_filename
- class TestVirtualPrinterManager:
- """Tests for VirtualPrinterManager orchestrator."""
- @pytest.fixture
- def manager(self):
- """Create a VirtualPrinterManager instance."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterManager
- return VirtualPrinterManager()
- def test_manager_starts_empty(self, manager):
- """Verify manager starts with no instances."""
- assert len(manager._instances) == 0
- assert manager.is_enabled is False
- def test_manager_get_status_empty(self, manager):
- """Verify get_status returns disabled state when no instances."""
- status = manager.get_status()
- assert status["enabled"] is False
- assert status["running"] is False
- assert status["mode"] == "immediate"
- def test_manager_is_enabled_with_instance(self, manager, tmp_path):
- """Verify is_enabled is True when instances exist."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=1,
- name="Test",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- base_dir=tmp_path,
- )
- manager._instances[1] = inst
- assert manager.is_enabled is True
- @pytest.mark.asyncio
- async def test_manager_remove_instance_server(self, manager, tmp_path):
- """Verify remove_instance stops and removes a server-mode instance."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=1,
- name="Test",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- base_dir=tmp_path,
- )
- inst.stop_server = AsyncMock()
- manager._instances[1] = inst
- await manager.remove_instance(1)
- assert 1 not in manager._instances
- inst.stop_server.assert_called_once()
- @pytest.mark.asyncio
- async def test_manager_remove_instance_proxy(self, manager, tmp_path):
- """Verify remove_instance stops proxy-mode instance."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=2,
- name="Proxy",
- mode="proxy",
- model="C11",
- access_code="",
- serial_suffix="391800002",
- target_printer_ip="192.168.1.100",
- base_dir=tmp_path,
- )
- inst.stop_proxy = AsyncMock()
- manager._instances[2] = inst
- await manager.remove_instance(2)
- assert 2 not in manager._instances
- inst.stop_proxy.assert_called_once()
- def test_manager_get_status_with_instance(self, manager, tmp_path):
- """Verify legacy get_status returns first instance data."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=1,
- name="Bambuddy",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- base_dir=tmp_path,
- )
- mock_task = MagicMock(done=MagicMock(return_value=False))
- inst._tasks = [mock_task]
- inst._pending_files = {"file1.3mf": Path("/tmp/file1.3mf")} # nosec B108
- manager._instances[1] = inst
- status = manager.get_status()
- assert status["enabled"] is True
- assert status["running"] is True
- assert status["mode"] == "immediate"
- assert status["name"] == "Bambuddy"
- assert status["serial"] == "01S00A391800001"
- assert status["model"] == "C11"
- assert status["model_name"] == "P1P"
- assert status["pending_files"] == 1
- def test_manager_get_all_status(self, manager, tmp_path):
- """Verify get_all_status returns status for all instances."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- for i in range(1, 3):
- inst = VirtualPrinterInstance(
- vp_id=i,
- name=f"VP{i}",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix=f"39180000{i}",
- base_dir=tmp_path,
- )
- manager._instances[i] = inst
- statuses = manager.get_all_status()
- assert len(statuses) == 2
- assert statuses[0]["name"] == "VP1"
- assert statuses[1]["name"] == "VP2"
- @pytest.mark.asyncio
- async def test_manager_stop_all(self, manager, tmp_path):
- """Verify stop_all removes all instances."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- for i in range(1, 3):
- inst = VirtualPrinterInstance(
- vp_id=i,
- name=f"VP{i}",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix=f"39180000{i}",
- base_dir=tmp_path,
- )
- inst.stop_server = AsyncMock()
- manager._instances[i] = inst
- await manager.stop_all()
- assert len(manager._instances) == 0
- # ========================================================================
- # Tests for sync_from_db config change detection
- # ========================================================================
- def _make_db_vp(self, **overrides):
- """Create a mock VirtualPrinter DB object."""
- defaults = {
- "id": 1,
- "name": "TestVP",
- "enabled": True,
- "mode": "immediate",
- "model": "C11",
- "access_code": "12345678",
- "serial_suffix": "391800001",
- "bind_ip": "",
- "remote_interface_ip": "",
- "target_printer_id": None,
- "auto_dispatch": True,
- "tailscale_disabled": True, # Opt-in default (#1070 UX fix)
- "position": 0,
- }
- defaults.update(overrides)
- vp = MagicMock()
- for k, v in defaults.items():
- setattr(vp, k, v)
- return vp
- def _setup_sync_mocks(self, manager, enabled_vps, tmp_path):
- """Wire up session_factory mock for sync_from_db."""
- mock_result = MagicMock()
- mock_result.scalars.return_value.all.return_value = enabled_vps
- mock_db = AsyncMock()
- mock_db.execute = AsyncMock(return_value=mock_result)
- mock_db.__aenter__ = AsyncMock(return_value=mock_db)
- mock_db.__aexit__ = AsyncMock(return_value=False)
- manager._session_factory = MagicMock(return_value=mock_db)
- manager._base_dir = tmp_path
- @pytest.mark.asyncio
- async def test_sync_from_db_restarts_on_mode_change(self, manager, tmp_path):
- """Verify sync_from_db restarts VP when mode changes."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=1,
- name="TestVP",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- base_dir=tmp_path,
- )
- inst.stop_server = AsyncMock()
- manager._instances[1] = inst
- # DB says mode changed to "archive"
- db_vp = self._make_db_vp(mode="archive")
- self._setup_sync_mocks(manager, [db_vp], tmp_path)
- with patch.object(manager, "remove_instance", new_callable=AsyncMock) as mock_remove:
- # Patch VirtualPrinterInstance to prevent actual start
- with patch("backend.app.services.virtual_printer.manager.VirtualPrinterInstance") as MockInst:
- mock_new = MagicMock()
- mock_new.start_server = AsyncMock()
- MockInst.return_value = mock_new
- await manager.sync_from_db()
- mock_remove.assert_called_once_with(1)
- @pytest.mark.asyncio
- async def test_sync_from_db_restarts_on_access_code_change(self, manager, tmp_path):
- """Verify sync_from_db restarts VP when access_code changes."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=1,
- name="TestVP",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- base_dir=tmp_path,
- )
- inst.stop_server = AsyncMock()
- manager._instances[1] = inst
- db_vp = self._make_db_vp(access_code="newcode99")
- self._setup_sync_mocks(manager, [db_vp], tmp_path)
- with patch.object(manager, "remove_instance", new_callable=AsyncMock) as mock_remove:
- with patch("backend.app.services.virtual_printer.manager.VirtualPrinterInstance") as MockInst:
- mock_new = MagicMock()
- mock_new.start_server = AsyncMock()
- MockInst.return_value = mock_new
- await manager.sync_from_db()
- mock_remove.assert_called_once_with(1)
- @pytest.mark.asyncio
- async def test_sync_from_db_skips_unchanged_instance(self, manager, tmp_path):
- """Verify sync_from_db does NOT restart when config is identical."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=1,
- name="TestVP",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- base_dir=tmp_path,
- )
- manager._instances[1] = inst
- # DB matches running config exactly
- db_vp = self._make_db_vp()
- self._setup_sync_mocks(manager, [db_vp], tmp_path)
- with patch.object(manager, "remove_instance", new_callable=AsyncMock) as mock_remove:
- await manager.sync_from_db()
- mock_remove.assert_not_called()
- @pytest.mark.asyncio
- async def test_sync_from_db_restarts_on_bind_ip_change(self, manager, tmp_path):
- """Verify sync_from_db restarts VP when bind_ip changes."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=1,
- name="TestVP",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- bind_ip="192.168.1.10",
- base_dir=tmp_path,
- )
- inst.stop_server = AsyncMock()
- manager._instances[1] = inst
- db_vp = self._make_db_vp(bind_ip="192.168.1.20")
- self._setup_sync_mocks(manager, [db_vp], tmp_path)
- with patch.object(manager, "remove_instance", new_callable=AsyncMock) as mock_remove:
- with patch("backend.app.services.virtual_printer.manager.VirtualPrinterInstance") as MockInst:
- mock_new = MagicMock()
- mock_new.start_server = AsyncMock()
- MockInst.return_value = mock_new
- await manager.sync_from_db()
- mock_remove.assert_called_once_with(1)
- @pytest.mark.asyncio
- async def test_sync_from_db_restarts_on_model_change(self, manager, tmp_path):
- """Verify sync_from_db restarts VP when model changes."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=1,
- name="TestVP",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- base_dir=tmp_path,
- )
- inst.stop_server = AsyncMock()
- manager._instances[1] = inst
- db_vp = self._make_db_vp(model="C12")
- self._setup_sync_mocks(manager, [db_vp], tmp_path)
- with patch.object(manager, "remove_instance", new_callable=AsyncMock) as mock_remove:
- with patch("backend.app.services.virtual_printer.manager.VirtualPrinterInstance") as MockInst:
- mock_new = MagicMock()
- mock_new.start_server = AsyncMock()
- MockInst.return_value = mock_new
- await manager.sync_from_db()
- mock_remove.assert_called_once_with(1)
- @pytest.mark.asyncio
- async def test_sync_from_db_does_not_restart_on_tailscale_toggle(self, manager, tmp_path):
- """Flipping tailscale_disabled is purely informational — must NOT trigger a restart.
- Cert provisioning was removed; the toggle only governs whether the VP card surfaces
- the host's Tailscale IP/FQDN to the user. No service needs to reload, so changing
- it through sync_from_db should leave any running instance untouched.
- """
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=1,
- name="TestVP",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800001",
- tailscale_disabled=False,
- base_dir=tmp_path,
- )
- inst.stop_server = AsyncMock()
- manager._instances[1] = inst
- db_vp = self._make_db_vp(tailscale_disabled=True)
- self._setup_sync_mocks(manager, [db_vp], tmp_path)
- with patch.object(manager, "remove_instance", new_callable=AsyncMock) as mock_remove:
- await manager.sync_from_db()
- mock_remove.assert_not_called()
- class TestFTPSession:
- """Tests for FTP session handling."""
- @pytest.fixture
- def mock_reader(self):
- """Create a mock StreamReader."""
- reader = AsyncMock()
- return reader
- @pytest.fixture
- def mock_writer(self):
- """Create a mock StreamWriter."""
- writer = MagicMock()
- writer.get_extra_info = MagicMock(return_value=("192.168.1.100", 12345))
- writer.write = MagicMock()
- writer.drain = AsyncMock()
- writer.close = MagicMock()
- writer.wait_closed = AsyncMock()
- writer.is_closing = MagicMock(return_value=False)
- return writer
- @pytest.fixture
- def ssl_context(self):
- """Create a mock SSL context."""
- return MagicMock()
- @pytest.fixture
- def session(self, mock_reader, mock_writer, ssl_context, tmp_path):
- """Create an FTPSession instance."""
- from backend.app.services.virtual_printer.ftp_server import FTPSession
- return FTPSession(
- reader=mock_reader,
- writer=mock_writer,
- upload_dir=tmp_path,
- access_code="12345678",
- ssl_context=ssl_context,
- on_file_received=None,
- )
- # ========================================================================
- # Tests for authentication
- # ========================================================================
- @pytest.mark.asyncio
- async def test_user_command_accepts_bblp(self, session):
- """Verify USER command accepts bblp user."""
- await session.cmd_USER("bblp")
- assert session.username == "bblp"
- @pytest.mark.asyncio
- async def test_pass_command_authenticates(self, session):
- """Verify PASS command authenticates with correct code."""
- session.username = "bblp"
- await session.cmd_PASS("12345678")
- assert session.authenticated is True
- @pytest.mark.asyncio
- async def test_pass_command_rejects_wrong_code(self, session):
- """Verify PASS command rejects wrong access code."""
- session.username = "bblp"
- await session.cmd_PASS("wrongcode")
- assert session.authenticated is False
- # ========================================================================
- # Tests for FTP commands
- # ========================================================================
- @pytest.mark.asyncio
- async def test_syst_command(self, session):
- """Verify SYST returns UNIX type."""
- await session.cmd_SYST("")
- session.writer.write.assert_called()
- call_args = session.writer.write.call_args[0][0].decode()
- assert "215" in call_args
- assert "UNIX" in call_args
- @pytest.mark.asyncio
- async def test_pwd_command_requires_auth(self, session):
- """Verify PWD requires authentication."""
- session.authenticated = False
- await session.cmd_PWD("")
- call_args = session.writer.write.call_args[0][0].decode()
- assert "530" in call_args
- @pytest.mark.asyncio
- async def test_pwd_command_when_authenticated(self, session):
- """Verify PWD returns root directory when authenticated."""
- session.authenticated = True
- await session.cmd_PWD("")
- call_args = session.writer.write.call_args[0][0].decode()
- assert "257" in call_args
- @pytest.mark.asyncio
- async def test_type_command_sets_binary(self, session):
- """Verify TYPE I sets binary mode."""
- session.authenticated = True
- await session.cmd_TYPE("I")
- assert session.transfer_type == "I"
- @pytest.mark.asyncio
- async def test_pbsz_command(self, session):
- """Verify PBSZ returns success."""
- await session.cmd_PBSZ("0")
- call_args = session.writer.write.call_args[0][0].decode()
- assert "200" in call_args
- @pytest.mark.asyncio
- async def test_prot_command_accepts_p(self, session):
- """Verify PROT P is accepted."""
- await session.cmd_PROT("P")
- call_args = session.writer.write.call_args[0][0].decode()
- assert "200" in call_args
- @pytest.mark.asyncio
- async def test_quit_command(self, session):
- """Verify QUIT sends goodbye and raises CancelledError."""
- with pytest.raises(asyncio.CancelledError):
- await session.cmd_QUIT("")
- class TestSSDPServer:
- """Tests for Virtual Printer SSDP server."""
- @pytest.fixture
- def ssdp_server(self):
- """Create a VirtualPrinterSSDPServer instance."""
- from backend.app.services.virtual_printer.ssdp_server import VirtualPrinterSSDPServer
- return VirtualPrinterSSDPServer(
- serial="TEST123",
- name="TestPrinter",
- model="BL-P001",
- )
- # ========================================================================
- # Tests for SSDP response
- # ========================================================================
- def test_build_notify_message(self, ssdp_server):
- """Verify NOTIFY packet contains required headers."""
- # Set a known IP for testing
- ssdp_server._local_ip = "192.168.1.100"
- message = ssdp_server._build_notify_message()
- assert b"NOTIFY" in message
- assert b"DevName.bambu.com: TestPrinter" in message
- assert b"USN: TEST123" in message
- def test_build_response_message(self, ssdp_server):
- """Verify response packet contains required headers."""
- # Set a known IP for testing
- ssdp_server._local_ip = "192.168.1.100"
- message = ssdp_server._build_response_message()
- assert b"HTTP/1.1 200 OK" in message
- assert b"DevName.bambu.com: TestPrinter" in message
- assert b"USN: TEST123" in message
- def test_ssdp_server_uses_correct_model(self, ssdp_server):
- """Verify SSDP server uses the provided model."""
- ssdp_server._local_ip = "192.168.1.100"
- message = ssdp_server._build_notify_message()
- assert b"DevModel.bambu.com: BL-P001" in message
- # ========================================================================
- # Tests for advertise_ip parameter
- # ========================================================================
- def test_advertise_ip_sets_local_ip(self):
- """Verify advertise_ip overrides auto-detection."""
- from backend.app.services.virtual_printer.ssdp_server import VirtualPrinterSSDPServer
- server = VirtualPrinterSSDPServer(
- serial="TEST123",
- name="TestPrinter",
- model="BL-P001",
- advertise_ip="10.0.0.50",
- )
- assert server._local_ip == "10.0.0.50"
- def test_advertise_ip_empty_string_uses_auto_detect(self):
- """Verify empty advertise_ip falls back to auto-detection."""
- from backend.app.services.virtual_printer.ssdp_server import VirtualPrinterSSDPServer
- server = VirtualPrinterSSDPServer(
- serial="TEST123",
- name="TestPrinter",
- model="BL-P001",
- advertise_ip="",
- )
- assert server._local_ip is None
- def test_advertise_ip_in_notify_message(self):
- """Verify NOTIFY message uses the advertise_ip."""
- from backend.app.services.virtual_printer.ssdp_server import VirtualPrinterSSDPServer
- server = VirtualPrinterSSDPServer(
- serial="TEST123",
- name="TestPrinter",
- model="BL-P001",
- advertise_ip="10.0.0.50",
- )
- message = server._build_notify_message()
- assert b"Location: 10.0.0.50" in message
- def test_advertise_ip_in_response_message(self):
- """Verify M-SEARCH response uses the advertise_ip."""
- from backend.app.services.virtual_printer.ssdp_server import VirtualPrinterSSDPServer
- server = VirtualPrinterSSDPServer(
- serial="TEST123",
- name="TestPrinter",
- model="BL-P001",
- advertise_ip="10.0.0.50",
- )
- message = server._build_response_message()
- assert b"Location: 10.0.0.50" in message
- def test_default_no_advertise_ip(self):
- """Verify default constructor has None local_ip (auto-detect)."""
- from backend.app.services.virtual_printer.ssdp_server import VirtualPrinterSSDPServer
- server = VirtualPrinterSSDPServer()
- assert server._local_ip is None
- class TestCertificateService:
- """Tests for TLS certificate generation."""
- @pytest.fixture
- def cert_service(self, tmp_path):
- """Create a CertificateService instance."""
- from backend.app.services.virtual_printer.certificate import CertificateService
- return CertificateService(cert_dir=tmp_path, serial="TEST123")
- def test_generate_certificates(self, cert_service, tmp_path):
- """Verify certificates are generated correctly."""
- cert_path, key_path = cert_service.generate_certificates()
- assert cert_path.exists()
- assert key_path.exists()
- # Verify certificate content
- cert_content = cert_path.read_text()
- assert "BEGIN CERTIFICATE" in cert_content
- key_content = key_path.read_text()
- assert "BEGIN" in key_content and "KEY" in key_content
- def test_certificates_reused_if_exist(self, cert_service):
- """Verify existing certificates are reused."""
- # First generation
- cert_path1, key_path1 = cert_service.generate_certificates()
- mtime1 = cert_path1.stat().st_mtime
- # Second call should reuse (via ensure_certificates)
- cert_path2, key_path2 = cert_service.ensure_certificates()
- mtime2 = cert_path2.stat().st_mtime
- assert mtime1 == mtime2 # File wasn't regenerated
- def test_delete_certificates(self, cert_service):
- """Verify certificates can be deleted."""
- cert_service.generate_certificates()
- assert cert_service.cert_path.exists()
- assert cert_service.key_path.exists()
- cert_service.delete_certificates()
- assert not cert_service.cert_path.exists()
- assert not cert_service.key_path.exists()
- def test_ensure_creates_if_not_exist(self, cert_service):
- """Verify ensure_certificates generates if not existing."""
- assert not cert_service.cert_path.exists()
- cert_path, key_path = cert_service.ensure_certificates()
- assert cert_path.exists()
- assert key_path.exists()
- class TestBindServer:
- """Tests for BindServer (port 3002 bind/detect protocol)."""
- @pytest.fixture
- def bind_server(self):
- """Create a BindServer instance."""
- from backend.app.services.virtual_printer.bind_server import BindServer
- return BindServer(
- serial="09400A391800001",
- model="O1D",
- name="Bambuddy",
- )
- def test_build_frame(self, bind_server):
- """Verify frame building produces correct format."""
- payload = {"login": {"command": "detect"}}
- frame = bind_server._build_frame(payload)
- # Header: 0xA5A5
- assert frame[:2] == b"\xa5\xa5"
- # Trailer: 0xA7A7
- assert frame[-2:] == b"\xa7\xa7"
- # Length field is total message size (LE uint16)
- import struct
- total_len = struct.unpack_from("<H", frame, 2)[0]
- assert total_len == len(frame)
- # JSON payload is between header and trailer
- import json
- json_bytes = frame[4:-2]
- parsed = json.loads(json_bytes)
- assert parsed == payload
- def test_parse_frame_valid(self, bind_server):
- """Verify valid frame parsing extracts JSON correctly."""
- import json
- import struct
- payload = {"login": {"command": "detect", "sequence_id": "20000"}}
- json_bytes = json.dumps(payload, separators=(",", ":")).encode()
- total_len = 4 + len(json_bytes) + 2
- frame = b"\xa5\xa5" + struct.pack("<H", total_len) + json_bytes + b"\xa7\xa7"
- result = bind_server._parse_frame(frame)
- assert result is not None
- assert result["login"]["command"] == "detect"
- assert result["login"]["sequence_id"] == "20000"
- def test_parse_frame_invalid_header(self, bind_server):
- """Verify invalid header returns None."""
- result = bind_server._parse_frame(b"\xbb\xbb\x06\x00{}\xa7\xa7")
- assert result is None
- def test_parse_frame_invalid_trailer(self, bind_server):
- """Verify invalid trailer returns None."""
- result = bind_server._parse_frame(b"\xa5\xa5\x06\x00{}\xbb\xbb")
- assert result is None
- def test_parse_frame_too_short(self, bind_server):
- """Verify short data returns None."""
- result = bind_server._parse_frame(b"\xa5\xa5\x00")
- assert result is None
- def test_parse_frame_invalid_json(self, bind_server):
- """Verify invalid JSON returns None."""
- import struct
- bad_json = b"not json"
- total_len = 4 + len(bad_json) + 2
- frame = b"\xa5\xa5" + struct.pack("<H", total_len) + bad_json + b"\xa7\xa7"
- result = bind_server._parse_frame(frame)
- assert result is None
- def test_build_frame_roundtrip(self, bind_server):
- """Verify build_frame output can be parsed back."""
- payload = {
- "login": {
- "bind": "free",
- "command": "detect",
- "connect": "lan",
- "dev_cap": 1,
- "id": "09400A391800001",
- "model": "O1D",
- "name": "Bambuddy",
- "sequence_id": 3021,
- "version": "01.00.00.00",
- }
- }
- frame = bind_server._build_frame(payload)
- parsed = bind_server._parse_frame(frame)
- assert parsed is not None
- assert parsed["login"]["id"] == "09400A391800001"
- assert parsed["login"]["model"] == "O1D"
- assert parsed["login"]["name"] == "Bambuddy"
- assert parsed["login"]["bind"] == "free"
- def test_bind_server_stores_config(self, bind_server):
- """Verify bind server stores serial, model, name."""
- assert bind_server.serial == "09400A391800001"
- assert bind_server.model == "O1D"
- assert bind_server.name == "Bambuddy"
- assert bind_server.version == "01.00.00.00"
- def test_bind_server_custom_version(self):
- """Verify custom firmware version is stored."""
- from backend.app.services.virtual_printer.bind_server import BindServer
- server = BindServer(
- serial="TEST123",
- model="C13",
- name="Test",
- version="02.03.04.05",
- )
- assert server.version == "02.03.04.05"
- def test_bind_ports_constant(self):
- """Verify BIND_PORTS includes both 3000 and 3002 for slicer compatibility."""
- from backend.app.services.virtual_printer.bind_server import BIND_PORTS
- assert 3000 in BIND_PORTS
- assert 3002 in BIND_PORTS
- def test_bind_server_initializes_empty_servers_list(self, bind_server):
- """Verify bind server starts with empty servers list."""
- assert bind_server._servers == []
- assert bind_server._running is False
- class TestSlicerProxyManager:
- """Tests for SlicerProxyManager (proxy mode)."""
- @pytest.fixture
- def proxy_manager(self, tmp_path):
- """Create a SlicerProxyManager instance."""
- from backend.app.services.virtual_printer.tcp_proxy import SlicerProxyManager
- # Create dummy cert files
- cert_path = tmp_path / "cert.pem"
- key_path = tmp_path / "key.pem"
- cert_path.write_text("-----BEGIN CERTIFICATE-----\ntest\n-----END CERTIFICATE-----")
- # Split string to avoid pre-commit hook false positive on test data
- key_path.write_text("-----BEGIN " + "PRIVATE KEY-----\ntest\n-----END " + "PRIVATE KEY-----")
- return SlicerProxyManager(
- target_host="192.168.1.100",
- cert_path=cert_path,
- key_path=key_path,
- )
- def test_proxy_manager_initializes_ports(self, proxy_manager):
- """Verify proxy manager has correct port constants."""
- # FTP proxy uses privileged port 990 to match what Bambu Studio expects
- assert proxy_manager.LOCAL_FTP_PORT == 990
- assert proxy_manager.LOCAL_MQTT_PORT == 8883
- assert proxy_manager.PRINTER_FTP_PORT == 990
- assert proxy_manager.PRINTER_MQTT_PORT == 8883
- assert proxy_manager.PRINTER_FILE_TRANSFER_PORT == 6000
- assert proxy_manager.PRINTER_RTSP_PORT == 322
- # Auxiliary ports: undocumented proprietary ports for A1/P1S etc.
- assert proxy_manager.PRINTER_AUX_PORTS == [2024, 2025, 2026]
- # Bind ports: both 3000 and 3002 for slicer compatibility
- assert proxy_manager.PRINTER_BIND_PORTS == [3000, 3002]
- # FTP data port range for transparent EPSV proxying
- assert proxy_manager.FTP_DATA_PORT_MIN == 50000
- assert proxy_manager.FTP_DATA_PORT_MAX == 50100
- def test_proxy_manager_stores_target_host(self, proxy_manager):
- """Verify proxy manager stores target host."""
- assert proxy_manager.target_host == "192.168.1.100"
- def test_get_status_before_start(self, proxy_manager):
- """Verify get_status returns zeros before start."""
- status = proxy_manager.get_status()
- assert status["running"] is False
- assert status["ftp_connections"] == 0
- assert status["mqtt_connections"] == 0
- @pytest.mark.asyncio
- async def test_proxy_start_creates_transparent_proxies(self, tmp_path):
- """Verify start() uses TCPProxy for FTP/FileTransfer/RTSP and TLSProxy only for MQTT.
- The transparent proxy architecture preserves end-to-end TLS between
- slicer and printer for all protocols except MQTT, which must be
- TLS-terminated to rewrite the printer's IP in MQTT payloads.
- """
- from unittest.mock import AsyncMock, patch
- from backend.app.services.virtual_printer.tcp_proxy import (
- SlicerProxyManager,
- TCPProxy,
- TLSProxy,
- )
- cert_path = tmp_path / "cert.pem"
- key_path = tmp_path / "key.pem"
- cert_path.write_text("-----BEGIN CERTIFICATE-----\ntest\n-----END CERTIFICATE-----")
- key_path.write_text("-----BEGIN " + "PRIVATE KEY-----\ntest\n-----END " + "PRIVATE KEY-----")
- mgr = SlicerProxyManager(
- target_host="192.168.1.100",
- cert_path=cert_path,
- key_path=key_path,
- bind_address="10.0.0.1",
- )
- # Mock asyncio.create_task and asyncio.gather to prevent actual server start
- with (
- patch("asyncio.create_task") as mock_create_task,
- patch("asyncio.gather", new_callable=AsyncMock),
- patch.object(SlicerProxyManager, "_log_activity"),
- ):
- mock_create_task.return_value = MagicMock()
- # start() will create proxies then try to gather tasks — we just
- # need to verify the proxy types after creation.
- # Trigger start but let gather return immediately.
- await mgr.start()
- # FTP, FileTransfer, RTSP should be TCPProxy (transparent)
- assert isinstance(mgr._ftp_proxy, TCPProxy), "FTP should be TCPProxy (transparent)"
- assert isinstance(mgr._file_transfer_proxy, TCPProxy), "FileTransfer should be TCPProxy"
- assert isinstance(mgr._rtsp_proxy, TCPProxy), "RTSP should be TCPProxy"
- # MQTT should be TLSProxy (TLS-terminated for IP rewriting)
- assert isinstance(mgr._mqtt_proxy, TLSProxy), "MQTT should be TLSProxy (TLS-terminated)"
- # Auxiliary ports (2024-2026) should be TCPProxy (transparent)
- assert len(mgr._aux_proxies) == 3, "Should have 3 aux port proxies"
- for ap in mgr._aux_proxies:
- assert isinstance(ap, TCPProxy), "Aux proxies should be TCPProxy"
- assert mgr._aux_proxies[0].listen_port == 2024
- assert mgr._aux_proxies[0].target_port == 2024
- assert mgr._aux_proxies[2].listen_port == 2026
- # FTP data ports should be pre-created as TCPProxy instances
- assert len(mgr._ftp_data_proxies) == 101 # 50000-50100 inclusive
- for dp in mgr._ftp_data_proxies:
- assert isinstance(dp, TCPProxy), "FTP data proxies should be TCPProxy"
- # Verify FTP data proxies target the same port on the printer
- first_dp = mgr._ftp_data_proxies[0]
- assert first_dp.listen_port == 50000
- assert first_dp.target_port == 50000
- assert first_dp.target_host == "192.168.1.100"
- last_dp = mgr._ftp_data_proxies[-1]
- assert last_dp.listen_port == 50100
- assert last_dp.target_port == 50100
- def test_proxy_manager_mqtt_has_ip_rewriting(self, tmp_path):
- """Verify MQTT proxy is configured with IP rewriting when bind_address is set."""
- from backend.app.services.virtual_printer.tcp_proxy import SlicerProxyManager
- cert_path = tmp_path / "cert.pem"
- key_path = tmp_path / "key.pem"
- cert_path.write_text("-----BEGIN CERTIFICATE-----\ntest\n-----END CERTIFICATE-----")
- key_path.write_text("-----BEGIN " + "PRIVATE KEY-----\ntest\n-----END " + "PRIVATE KEY-----")
- mgr = SlicerProxyManager(
- target_host="192.168.1.100",
- cert_path=cert_path,
- key_path=key_path,
- bind_address="10.0.0.1",
- )
- # Before start, proxies are None — verify constructor stores rewrite config
- assert mgr.bind_address == "10.0.0.1"
- assert mgr.target_host == "192.168.1.100"
- class TestSSDPProxy:
- """Tests for SSDPProxy (cross-network SSDP relay)."""
- @pytest.fixture
- def ssdp_proxy(self):
- """Create an SSDPProxy instance."""
- from backend.app.services.virtual_printer.ssdp_server import SSDPProxy
- return SSDPProxy(
- local_interface_ip="192.168.1.100",
- remote_interface_ip="10.0.0.100",
- target_printer_ip="192.168.1.50",
- )
- def test_ssdp_proxy_stores_interface_ips(self, ssdp_proxy):
- """Verify SSDPProxy stores interface IPs correctly."""
- assert ssdp_proxy.local_interface_ip == "192.168.1.100"
- assert ssdp_proxy.remote_interface_ip == "10.0.0.100"
- assert ssdp_proxy.target_printer_ip == "192.168.1.50"
- def test_rewrite_ssdp_location(self, ssdp_proxy):
- """Verify SSDP Location header is rewritten to remote interface IP."""
- original_packet = b"NOTIFY * HTTP/1.1\r\nLocation: 192.168.1.50\r\nDevName.bambu.com: TestPrinter\r\n\r\n"
- rewritten = ssdp_proxy._rewrite_ssdp(original_packet)
- # Location should be changed to remote interface IP
- assert b"Location: 10.0.0.100" in rewritten
- assert b"Location: 192.168.1.50" not in rewritten
- # Other headers should be preserved
- assert b"DevName.bambu.com: TestPrinter" in rewritten
- def test_rewrite_ssdp_location_case_insensitive(self, ssdp_proxy):
- """Verify SSDP Location rewrite is case insensitive."""
- original_packet = b"NOTIFY * HTTP/1.1\r\nlocation: 192.168.1.50\r\n\r\n"
- rewritten = ssdp_proxy._rewrite_ssdp(original_packet)
- assert b"10.0.0.100" in rewritten
- def test_rewrite_ssdp_location_no_match(self, ssdp_proxy):
- """Verify packet without Location header is returned unchanged."""
- original_packet = b"NOTIFY * HTTP/1.1\r\nDevName.bambu.com: Test\r\n\r\n"
- rewritten = ssdp_proxy._rewrite_ssdp(original_packet)
- # No Location header, but _rewrite_ssdp logs a warning and returns as-is
- assert b"DevName.bambu.com: Test" in rewritten
- def test_parse_ssdp_message(self, ssdp_proxy):
- """Verify SSDP message parsing extracts headers."""
- packet = (
- b"NOTIFY * HTTP/1.1\r\n"
- b"Location: 192.168.1.50\r\n"
- b"DevName.bambu.com: TestPrinter\r\n"
- b"DevModel.bambu.com: BL-P001\r\n"
- b"\r\n"
- )
- headers = ssdp_proxy._parse_ssdp_message(packet)
- assert headers["location"] == "192.168.1.50"
- assert headers["devname.bambu.com"] == "TestPrinter"
- assert headers["devmodel.bambu.com"] == "BL-P001"
- class TestVirtualPrinterManagerDirectories:
- """Tests for VirtualPrinterManager directory management."""
- def test_ensure_base_directories_creates_subdirs(self, tmp_path):
- """Verify _ensure_base_directories creates required base directories."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterManager
- manager = VirtualPrinterManager()
- manager._base_dir = tmp_path / "virtual_printer"
- manager._ensure_base_directories()
- assert (tmp_path / "virtual_printer").exists()
- assert (tmp_path / "virtual_printer" / "uploads").exists()
- assert (tmp_path / "virtual_printer" / "certs").exists()
- def test_ensure_base_directories_handles_permission_error(self, tmp_path, caplog):
- """Verify _ensure_base_directories logs error on permission failure."""
- import logging
- from backend.app.services.virtual_printer.manager import VirtualPrinterManager
- manager = VirtualPrinterManager()
- vp_dir = tmp_path / "virtual_printer"
- manager._base_dir = vp_dir
- original_mkdir = type(vp_dir).mkdir
- def mock_mkdir(self, *args, **kwargs):
- if "virtual_printer" in str(self):
- raise PermissionError("Permission denied")
- return original_mkdir(self, *args, **kwargs)
- with caplog.at_level(logging.ERROR), patch.object(type(vp_dir), "mkdir", mock_mkdir):
- manager._ensure_base_directories()
- assert "Permission denied" in caplog.text
- def test_instance_creates_per_vp_directories(self, tmp_path):
- """Verify VirtualPrinterInstance creates per-VP upload and cert dirs."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- VirtualPrinterInstance(
- vp_id=42,
- name="Test",
- mode="immediate",
- model="C11",
- access_code="12345678",
- serial_suffix="391800042",
- base_dir=tmp_path,
- )
- assert (tmp_path / "uploads" / "42").exists()
- assert (tmp_path / "uploads" / "42" / "cache").exists()
- assert (tmp_path / "certs" / "42").exists()
- class TestVirtualPrinterInstanceProxyMode:
- """Tests for VirtualPrinterInstance proxy mode."""
- @pytest.fixture
- def proxy_instance(self, tmp_path):
- """Create a proxy-mode VirtualPrinterInstance."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- return VirtualPrinterInstance(
- vp_id=10,
- name="ProxyTest",
- mode="proxy",
- model="C11",
- access_code="",
- serial_suffix="391800010",
- target_printer_ip="192.168.1.100",
- target_printer_serial="01P00A000000001",
- base_dir=tmp_path,
- )
- def test_proxy_instance_properties(self, proxy_instance):
- """Verify proxy instance stores config correctly."""
- assert proxy_instance.is_proxy is True
- assert proxy_instance.mode == "proxy"
- assert proxy_instance.target_printer_ip == "192.168.1.100"
- assert proxy_instance.target_printer_serial == "01P00A000000001"
- def test_proxy_instance_does_not_require_access_code(self, proxy_instance):
- """Verify proxy mode can have empty access code."""
- assert proxy_instance.access_code == ""
- def test_get_status_proxy_includes_proxy_fields(self, proxy_instance):
- """Verify get_status includes proxy fields when proxy is active."""
- mock_proxy = MagicMock()
- mock_proxy.get_status.return_value = {
- "running": True,
- "ftp_port": 990,
- "mqtt_port": 8883,
- "ftp_connections": 1,
- "mqtt_connections": 2,
- "target_host": "192.168.1.100",
- }
- proxy_instance._proxy = mock_proxy
- status = proxy_instance.get_status()
- assert "proxy" in status
- assert status["proxy"]["ftp_port"] == 990
- assert status["proxy"]["mqtt_connections"] == 2
- def test_proxy_instance_stores_remote_interface(self, tmp_path):
- """Verify proxy instance stores remote_interface_ip."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=11,
- name="Proxy2",
- mode="proxy",
- model="C11",
- access_code="",
- serial_suffix="391800011",
- target_printer_ip="192.168.1.100",
- remote_interface_ip="10.0.0.50",
- base_dir=tmp_path,
- )
- assert inst.remote_interface_ip == "10.0.0.50"
- class TestVirtualPrinterInstanceIPOverride:
- """Tests for remote_interface_ip and bind_ip on VirtualPrinterInstance."""
- @pytest.fixture
- def instance_with_remote_ip(self, tmp_path):
- """Create an instance with remote_interface_ip set."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- return VirtualPrinterInstance(
- vp_id=20,
- name="IPTest",
- mode="immediate",
- model="BL-P001",
- access_code="12345678",
- serial_suffix="391800020",
- bind_ip="192.168.1.50",
- remote_interface_ip="10.0.0.50",
- base_dir=tmp_path,
- )
- def test_instance_stores_bind_ip(self, instance_with_remote_ip):
- """Verify bind_ip is stored."""
- assert instance_with_remote_ip.bind_ip == "192.168.1.50"
- def test_instance_stores_remote_interface_ip(self, instance_with_remote_ip):
- """Verify remote_interface_ip is stored."""
- assert instance_with_remote_ip.remote_interface_ip == "10.0.0.50"
- def test_generate_certificates_includes_remote_and_bind_ip(self, instance_with_remote_ip):
- """Verify generate_certificates passes remote_interface_ip and bind_ip as SANs."""
- with (
- patch.object(instance_with_remote_ip._cert_service, "delete_printer_certificate"),
- patch.object(
- instance_with_remote_ip._cert_service,
- "generate_certificates",
- return_value=(Path("/tmp/cert.pem"), Path("/tmp/key.pem")), # nosec B108
- ) as mock_gen,
- ):
- instance_with_remote_ip.generate_certificates()
- mock_gen.assert_called_once_with(additional_ips=["10.0.0.50", "192.168.1.50"])
- def test_generate_certificates_no_remote_ip(self, tmp_path):
- """Verify generate_certificates passes only bind_ip when no remote_interface_ip."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=21,
- name="NoRemote",
- mode="immediate",
- model="BL-P001",
- access_code="12345678",
- serial_suffix="391800021",
- bind_ip="192.168.1.50",
- base_dir=tmp_path,
- )
- with (
- patch.object(inst._cert_service, "delete_printer_certificate"),
- patch.object(
- inst._cert_service,
- "generate_certificates",
- return_value=(Path("/tmp/cert.pem"), Path("/tmp/key.pem")), # nosec B108
- ) as mock_gen,
- ):
- inst.generate_certificates()
- mock_gen.assert_called_once_with(additional_ips=["192.168.1.50"])
- def test_generate_certificates_no_ips(self, tmp_path):
- """Verify generate_certificates passes None when no IPs configured."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=22,
- name="NoIPs",
- mode="immediate",
- model="BL-P001",
- access_code="12345678",
- serial_suffix="391800022",
- base_dir=tmp_path,
- )
- with (
- patch.object(inst._cert_service, "delete_printer_certificate"),
- patch.object(
- inst._cert_service,
- "generate_certificates",
- return_value=(Path("/tmp/cert.pem"), Path("/tmp/key.pem")), # nosec B108
- ) as mock_gen,
- ):
- inst.generate_certificates()
- mock_gen.assert_called_once_with(additional_ips=None)
- class TestBindServer:
- """Tests for the BindServer (port 3002 bind/detect protocol)."""
- @pytest.fixture
- def bind_server(self):
- """Create a BindServer instance."""
- from backend.app.services.virtual_printer.bind_server import BindServer
- return BindServer(
- serial="01S00C000000001",
- model="BL-P001",
- name="Bambuddy",
- )
- def test_build_frame(self, bind_server):
- """Verify frame format: 0xA5A5 + len(u16le) + JSON + 0xA7A7."""
- payload = {"login": {"command": "detect"}}
- frame = bind_server._build_frame(payload)
- assert frame[:2] == b"\xa5\xa5"
- assert frame[-2:] == b"\xa7\xa7"
- # Length field is total message size
- import struct
- total_len = struct.unpack_from("<H", frame, 2)[0]
- assert total_len == len(frame)
- # JSON payload is between header and trailer
- import json
- json_bytes = frame[4:-2]
- parsed = json.loads(json_bytes)
- assert parsed == payload
- def test_parse_frame_valid(self, bind_server):
- """Verify valid frame parsing."""
- frame = bind_server._build_frame({"login": {"command": "detect", "sequence_id": "20000"}})
- result = bind_server._parse_frame(frame)
- assert result is not None
- assert result["login"]["command"] == "detect"
- assert result["login"]["sequence_id"] == "20000"
- def test_parse_frame_invalid_header(self, bind_server):
- """Verify invalid header returns None."""
- frame = b"\xb5\xb5\x10\x00" + b'{"login":{}}' + b"\xa7\xa7"
- assert bind_server._parse_frame(frame) is None
- def test_parse_frame_invalid_trailer(self, bind_server):
- """Verify invalid trailer returns None."""
- frame = b"\xa5\xa5\x10\x00" + b'{"login":{}}' + b"\xb7\xb7"
- assert bind_server._parse_frame(frame) is None
- def test_parse_frame_too_short(self, bind_server):
- """Verify short data returns None."""
- assert bind_server._parse_frame(b"\xa5\xa5\x00") is None
- assert bind_server._parse_frame(b"") is None
- def test_parse_frame_invalid_json(self, bind_server):
- """Verify invalid JSON returns None."""
- import struct
- bad_json = b"not json"
- total_len = 4 + len(bad_json) + 2
- frame = b"\xa5\xa5" + struct.pack("<H", total_len) + bad_json + b"\xa7\xa7"
- assert bind_server._parse_frame(frame) is None
- def test_build_frame_roundtrip(self, bind_server):
- """Verify build then parse roundtrip."""
- original = {"login": {"bind": "free", "command": "detect", "id": "01S00C000000001"}}
- frame = bind_server._build_frame(original)
- parsed = bind_server._parse_frame(frame)
- assert parsed == original
- def test_bind_server_stores_config(self, bind_server):
- """Verify config is stored correctly."""
- assert bind_server.serial == "01S00C000000001"
- assert bind_server.model == "BL-P001"
- assert bind_server.name == "Bambuddy"
- assert bind_server.version == "01.00.00.00"
- def test_bind_server_custom_version(self):
- """Verify custom firmware version is stored."""
- from backend.app.services.virtual_printer.bind_server import BindServer
- server = BindServer(
- serial="01S00C000000001",
- model="BL-P001",
- name="Bambuddy",
- version="01.09.00.10",
- )
- assert server.version == "01.09.00.10"
- def test_bind_ports_includes_both(self):
- """Verify BIND_PORTS includes both 3000 and 3002 for slicer compatibility."""
- from backend.app.services.virtual_printer.bind_server import BIND_PORTS
- assert 3000 in BIND_PORTS
- assert 3002 in BIND_PORTS
- def test_bind_server_initializes_empty_servers_list(self, bind_server):
- """Verify bind server starts with empty servers list."""
- assert bind_server._servers == []
- assert bind_server._running is False
- @pytest.mark.asyncio
- async def test_start_server_creates_bind_server(self, tmp_path):
- """Verify start_server creates BindServer with correct params."""
- from backend.app.services.virtual_printer.manager import VirtualPrinterInstance
- inst = VirtualPrinterInstance(
- vp_id=99,
- name="Bambuddy",
- mode="immediate",
- model="BL-P001",
- access_code="12345678",
- serial_suffix="391800099",
- bind_ip="192.168.1.50",
- base_dir=tmp_path,
- )
- with (
- patch("backend.app.services.virtual_printer.manager.VirtualPrinterSSDPServer"),
- patch("backend.app.services.virtual_printer.manager.VirtualPrinterFTPServer"),
- patch("backend.app.services.virtual_printer.manager.SimpleMQTTServer"),
- patch("backend.app.services.virtual_printer.manager.BindServer") as mock_bind_cls,
- patch.object(inst._cert_service, "delete_printer_certificate"),
- patch.object(
- inst._cert_service,
- "generate_certificates",
- return_value=(Path("/tmp/cert.pem"), Path("/tmp/key.pem")), # nosec B108
- ),
- ):
- await inst.start_server()
- mock_bind_cls.assert_called_once_with(
- serial=inst.serial,
- model="BL-P001",
- name="Bambuddy",
- bind_address="192.168.1.50",
- cert_path=Path("/tmp/cert.pem"), # nosec B108
- key_path=Path("/tmp/key.pem"), # nosec B108
- )
- class TestResolveModelCodes:
- """Tests for model code resolution (display name → SSDP code)."""
- def test_display_name_to_model_code_maps_all_models(self):
- """Verify reverse mapping covers all VIRTUAL_PRINTER_MODELS entries."""
- from backend.app.services.virtual_printer.manager import DISPLAY_NAME_TO_MODEL_CODE, VIRTUAL_PRINTER_MODELS
- for _code, display_name in VIRTUAL_PRINTER_MODELS.items():
- assert display_name in DISPLAY_NAME_TO_MODEL_CODE
- # For non-duplicate display names, should map back to a valid code
- assert DISPLAY_NAME_TO_MODEL_CODE[display_name] in VIRTUAL_PRINTER_MODELS
- def test_resolve_printer_model_with_ssdp_code(self):
- """SSDP codes pass through unchanged."""
- from backend.app.api.routes.virtual_printers import _resolve_printer_model
- assert _resolve_printer_model("BL-P001") == "BL-P001"
- assert _resolve_printer_model("O1D") == "O1D"
- assert _resolve_printer_model("N2S") == "N2S"
- def test_resolve_printer_model_with_display_name(self):
- """Display names resolve to SSDP codes."""
- from backend.app.api.routes.virtual_printers import _resolve_printer_model
- assert _resolve_printer_model("X1C") == "BL-P001"
- assert _resolve_printer_model("H2D") == "O1D"
- assert _resolve_printer_model("A1") == "N2S"
- assert _resolve_printer_model("P1S") == "C12"
- def test_resolve_printer_model_with_none_or_unknown(self):
- """None and unknown values return None."""
- from backend.app.api.routes.virtual_printers import _resolve_printer_model
- assert _resolve_printer_model(None) is None
- assert _resolve_printer_model("UnknownModel") is None
- class TestMqttIpRewrite:
- """Tests for TLSProxy._rewrite_mqtt_ip() MQTT packet IP rewriting."""
- @staticmethod
- def _build_mqtt_publish(topic: str, payload: bytes) -> bytes:
- """Build a minimal MQTT PUBLISH packet."""
- # PUBLISH fixed header: type 3, no flags
- topic_bytes = topic.encode("utf-8")
- # Variable header: topic length (2 bytes) + topic
- var_header = len(topic_bytes).to_bytes(2, "big") + topic_bytes
- body = var_header + payload
- # Encode remaining length
- remaining = len(body)
- header = bytearray([0x30]) # PUBLISH, QoS 0
- while True:
- encoded_byte = remaining % 128
- remaining //= 128
- if remaining > 0:
- encoded_byte |= 0x80
- header.append(encoded_byte)
- if remaining == 0:
- break
- return bytes(header) + body
- @staticmethod
- def _build_mqtt_pingreq() -> bytes:
- """Build an MQTT PINGREQ packet (2 bytes, no payload)."""
- return b"\xc0\x00"
- def test_rewrite_ip_in_publish(self):
- """IP string in PUBLISH payload is rewritten."""
- from backend.app.services.virtual_printer.tcp_proxy import TLSProxy
- payload = b'{"rtsp_url":"rtsps://192.168.1.100:322/live"}'
- packet = self._build_mqtt_publish("device/status", payload)
- result, buf = TLSProxy._rewrite_mqtt_ip(packet, b"192.168.1.100", b"10.0.0.1", bytearray())
- assert b"10.0.0.1" in result
- assert b"192.168.1.100" not in result
- def test_no_rewrite_when_ip_absent(self):
- """Packets without the target IP are passed through unchanged."""
- from backend.app.services.virtual_printer.tcp_proxy import TLSProxy
- payload = b'{"status":"idle"}'
- packet = self._build_mqtt_publish("device/status", payload)
- result, buf = TLSProxy._rewrite_mqtt_ip(packet, b"192.168.1.100", b"10.0.0.1", bytearray())
- assert result == packet
- def test_non_publish_packets_unchanged(self):
- """Non-PUBLISH packets (e.g. PINGREQ) are never rewritten."""
- from backend.app.services.virtual_printer.tcp_proxy import TLSProxy
- pingreq = self._build_mqtt_pingreq()
- result, buf = TLSProxy._rewrite_mqtt_ip(pingreq, b"192.168.1.100", b"10.0.0.1", bytearray())
- assert result == pingreq
- def test_rewrite_preserves_packet_framing(self):
- """Rewritten packet has valid MQTT remaining length."""
- from backend.app.services.virtual_printer.tcp_proxy import TLSProxy
- # Use IPs of different lengths to test length re-encoding
- old_ip = b"192.168.255.133" # 15 bytes
- new_ip = b"10.0.0.1" # 8 bytes
- payload = b'{"ip":"192.168.255.133"}'
- packet = self._build_mqtt_publish("device/status", payload)
- result, buf = TLSProxy._rewrite_mqtt_ip(packet, old_ip, new_ip, bytearray())
- # Parse the result to verify framing
- assert result[0] == 0x30 # PUBLISH header byte
- # Decode remaining length
- pos = 1
- remaining = 0
- multiplier = 1
- while True:
- b = result[pos]
- pos += 1
- remaining += (b & 0x7F) * multiplier
- multiplier *= 128
- if (b & 0x80) == 0:
- break
- # Remaining length should match actual data
- assert pos + remaining == len(result)
- assert new_ip in result
- def test_incomplete_packet_buffered(self):
- """Incomplete packet at end of chunk is buffered for next call."""
- from backend.app.services.virtual_printer.tcp_proxy import TLSProxy
- payload = b'{"ip":"192.168.1.100"}'
- packet = self._build_mqtt_publish("device/status", payload)
- # Split packet in the middle
- half = len(packet) // 2
- chunk1 = packet[:half]
- chunk2 = packet[half:]
- result1, buf = TLSProxy._rewrite_mqtt_ip(chunk1, b"192.168.1.100", b"10.0.0.1", bytearray())
- # First chunk should be buffered (incomplete packet)
- assert len(buf) > 0
- result2, buf = TLSProxy._rewrite_mqtt_ip(chunk2, b"192.168.1.100", b"10.0.0.1", buf)
- # Second chunk completes the packet, IP should be rewritten
- combined = result1 + result2
- assert b"10.0.0.1" in combined
- assert b"192.168.1.100" not in combined
- def test_multiple_packets_in_one_chunk(self):
- """Multiple MQTT packets in a single chunk are all processed."""
- from backend.app.services.virtual_printer.tcp_proxy import TLSProxy
- payload1 = b'{"ip":"192.168.1.100"}'
- payload2 = b'{"other":"data"}'
- packet1 = self._build_mqtt_publish("topic1", payload1)
- packet2 = self._build_mqtt_publish("topic2", payload2)
- combined = packet1 + packet2
- result, buf = TLSProxy._rewrite_mqtt_ip(combined, b"192.168.1.100", b"10.0.0.1", bytearray())
- assert b"10.0.0.1" in result
- assert b"192.168.1.100" not in result
- # Second packet should still be present
- assert b"other" in result
- def test_extra_replacements(self):
- """Extra replacement pairs (e.g. integer IP) are also applied."""
- from backend.app.services.virtual_printer.tcp_proxy import TLSProxy
- payload = b'{"net":{"info":[{"ip":2248124608}]}}'
- packet = self._build_mqtt_publish("device/status", payload)
- result, buf = TLSProxy._rewrite_mqtt_ip(
- packet,
- b"NOMATCH",
- b"NOREPLACE",
- bytearray(),
- extra_replacements=[(b"2248124608", b"285190336")],
- )
- assert b"285190336" in result
- assert b"2248124608" not in result
- class TestIpToLeIntBytes:
- """Tests for TLSProxy._ip_to_le_int_bytes() integer IP conversion."""
- def test_converts_ip_to_le_int(self):
- from backend.app.services.virtual_printer.tcp_proxy import TLSProxy
- assert TLSProxy._ip_to_le_int_bytes("192.168.255.133") == b"2248124608"
- assert TLSProxy._ip_to_le_int_bytes("192.168.255.16") == b"285190336"
- assert TLSProxy._ip_to_le_int_bytes("10.0.0.1") == b"16777226"
- def test_roundtrip(self):
- """Verify the integer converts back to the correct IP."""
- import struct
- from backend.app.services.virtual_printer.tcp_proxy import TLSProxy
- for ip in ["192.168.1.1", "10.0.0.1", "172.16.0.100", "192.168.255.133"]:
- le_int = int(TLSProxy._ip_to_le_int_bytes(ip))
- parts = ip.split(".")
- expected = struct.unpack("<I", bytes(int(p) for p in parts))[0]
- assert le_int == expected
- class TestSSDPProxyName:
- """Tests for SSDPProxy VP name rewriting."""
- @pytest.fixture
- def ssdp_proxy_with_name(self):
- from backend.app.services.virtual_printer.ssdp_server import SSDPProxy
- return SSDPProxy(
- local_interface_ip="192.168.1.100",
- remote_interface_ip="10.0.0.100",
- target_printer_ip="192.168.1.50",
- name="H2D-1 Proxy",
- )
- @pytest.fixture
- def ssdp_proxy_without_name(self):
- from backend.app.services.virtual_printer.ssdp_server import SSDPProxy
- return SSDPProxy(
- local_interface_ip="192.168.1.100",
- remote_interface_ip="10.0.0.100",
- target_printer_ip="192.168.1.50",
- )
- def test_rewrite_uses_configured_name(self, ssdp_proxy_with_name):
- """When name is set, DevName is replaced entirely."""
- packet = b"NOTIFY * HTTP/1.1\r\nLocation: 192.168.1.50\r\nDevName.bambu.com: RealPrinter\r\nDevBind.bambu.com: cloud\r\n\r\n"
- rewritten = ssdp_proxy_with_name._rewrite_ssdp(packet)
- assert b"DevName.bambu.com: H2D-1 Proxy" in rewritten
- assert b"RealPrinter" not in rewritten
- def test_rewrite_appends_proxy_without_name(self, ssdp_proxy_without_name):
- """When no name is set, ' - Proxy' is appended to the real name."""
- packet = b"NOTIFY * HTTP/1.1\r\nLocation: 192.168.1.50\r\nDevName.bambu.com: RealPrinter\r\nDevBind.bambu.com: cloud\r\n\r\n"
- rewritten = ssdp_proxy_without_name._rewrite_ssdp(packet)
- assert b"DevName.bambu.com: RealPrinter - Proxy" in rewritten
|