| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328 |
- """Tests for the per-slice 3MF input normalisation helpers."""
- from __future__ import annotations
- import json
- import zipfile
- from io import BytesIO
- from backend.app.services.slicer_3mf_convert import (
- count_plates_in_3mf,
- extract_source_printer_model,
- merge_plate_3mfs,
- substitute_unused_plate_filaments,
- )
- def _make_3mf(entries: dict[str, bytes]) -> bytes:
- buf = BytesIO()
- with zipfile.ZipFile(buf, "w", zipfile.ZIP_DEFLATED) as zf:
- for name, payload in entries.items():
- zf.writestr(name, payload)
- return buf.getvalue()
- class TestExtractSourcePrinterModel:
- def test_returns_canonical_short_code_for_x1c(self):
- # Raw field is the long display name; we need the short code so
- # is_dual_nozzle_model() matches against the model registry.
- cfg = json.dumps({"printer_model": "Bambu Lab X1 Carbon", "other": "field"}).encode()
- zip_bytes = _make_3mf({"Metadata/project_settings.config": cfg})
- assert extract_source_printer_model(zip_bytes) == "X1C"
- def test_returns_canonical_short_code_for_h2d(self):
- cfg = json.dumps({"printer_model": "Bambu Lab H2D"}).encode()
- zip_bytes = _make_3mf({"Metadata/project_settings.config": cfg})
- assert extract_source_printer_model(zip_bytes) == "H2D"
- def test_dual_nozzle_check_works_on_extracted_code(self):
- """The whole point of canonicalising in this helper: the result
- must feed straight into is_dual_nozzle_model() without further
- normalisation."""
- from backend.app.utils.printer_models import is_dual_nozzle_model
- h2d = _make_3mf({"Metadata/project_settings.config": json.dumps({"printer_model": "Bambu Lab H2D"}).encode()})
- x1c = _make_3mf(
- {"Metadata/project_settings.config": json.dumps({"printer_model": "Bambu Lab X1 Carbon"}).encode()}
- )
- assert is_dual_nozzle_model(extract_source_printer_model(h2d)) is True
- assert is_dual_nozzle_model(extract_source_printer_model(x1c)) is False
- def test_returns_none_when_field_missing(self):
- cfg = json.dumps({"other": "field"}).encode()
- zip_bytes = _make_3mf({"Metadata/project_settings.config": cfg})
- assert extract_source_printer_model(zip_bytes) is None
- def test_returns_none_when_field_empty(self):
- cfg = json.dumps({"printer_model": ""}).encode()
- zip_bytes = _make_3mf({"Metadata/project_settings.config": cfg})
- assert extract_source_printer_model(zip_bytes) is None
- def test_returns_none_when_no_embedded_config(self):
- zip_bytes = _make_3mf({"Metadata/other.txt": b"hello"})
- assert extract_source_printer_model(zip_bytes) is None
- def test_returns_none_for_non_zip_bytes(self):
- assert extract_source_printer_model(b"not a zip") is None
- def test_returns_none_for_malformed_json(self):
- zip_bytes = _make_3mf({"Metadata/project_settings.config": b"{not json"})
- assert extract_source_printer_model(zip_bytes) is None
- def test_returns_none_when_config_is_list_not_dict(self):
- cfg = json.dumps(["not", "a", "dict"]).encode()
- zip_bytes = _make_3mf({"Metadata/project_settings.config": cfg})
- assert extract_source_printer_model(zip_bytes) is None
- class TestCountPlatesIn3mf:
- def test_counts_plater_id_entries(self):
- xml = (
- b'<?xml version="1.0"?>\n<config>\n'
- b'<plate><metadata key="plater_id" value="1"/></plate>\n'
- b'<plate><metadata key="plater_id" value="2"/></plate>\n'
- b'<plate><metadata key="plater_id" value="3"/></plate>\n'
- b"</config>\n"
- )
- zip_bytes = _make_3mf({"Metadata/model_settings.config": xml})
- assert count_plates_in_3mf(zip_bytes) == 3
- def test_returns_zero_for_no_model_settings(self):
- zip_bytes = _make_3mf({"3D/3dmodel.model": b"<model/>"})
- assert count_plates_in_3mf(zip_bytes) == 0
- def test_returns_zero_for_non_zip(self):
- assert count_plates_in_3mf(b"not a zip") == 0
- def test_returns_zero_when_no_plate_ids(self):
- zip_bytes = _make_3mf({"Metadata/model_settings.config": b"<config/>"})
- assert count_plates_in_3mf(zip_bytes) == 0
- class TestMergePlate3mfs:
- """Per-plate cross-class loop output → merged multi-plate 3MF. The
- merge needs to: (1) carry forward the first plate's base metadata
- (project_settings, model_settings, 3dmodel), (2) overlay each
- plate's gcode + thumbnails, (3) re-assemble slice_info.config to
- list every plate."""
- @staticmethod
- def _single_plate_3mf(plate_num: int, gcode_bytes: bytes, slice_info_block: str | None = None) -> bytes:
- slice_info = (
- '<?xml version="1.0" encoding="UTF-8"?>\n<config>\n'
- '<header><header_item key="X-BBL-Client-Type" value="slicer"/></header>\n'
- + (slice_info_block or f'<plate><metadata key="index" value="{plate_num}"/></plate>')
- + "\n</config>\n"
- ).encode("utf-8")
- return _make_3mf(
- {
- "3D/3dmodel.model": f"<model plate={plate_num}/>".encode(),
- "Metadata/project_settings.config": b'{"printer_model": "Bambu Lab H2D"}',
- "Metadata/model_settings.config": b"<config/>",
- "Metadata/slice_info.config": slice_info,
- f"Metadata/plate_{plate_num}.gcode": gcode_bytes,
- f"Metadata/plate_{plate_num}.gcode.md5": b"d41d8cd98f00b204e9800998ecf8427e",
- f"Metadata/plate_{plate_num}.json": b"{}",
- f"Metadata/plate_{plate_num}.png": b"PLATE_PNG",
- f"Metadata/plate_{plate_num}_small.png": b"SMALL",
- f"Metadata/top_{plate_num}.png": b"TOP",
- f"Metadata/pick_{plate_num}.png": b"PICK",
- }
- )
- def test_empty_input_raises(self):
- import pytest as _pytest
- with _pytest.raises(ValueError):
- merge_plate_3mfs([])
- def test_single_plate_is_passthrough(self):
- only = self._single_plate_3mf(1, b"GCODE-1")
- assert merge_plate_3mfs([(1, only)]) == only
- def test_overlays_per_plate_artifacts(self):
- p1 = self._single_plate_3mf(1, b"GCODE-PLATE-1")
- p2 = self._single_plate_3mf(2, b"GCODE-PLATE-2")
- p3 = self._single_plate_3mf(3, b"GCODE-PLATE-3")
- merged = merge_plate_3mfs([(1, p1), (2, p2), (3, p3)])
- with zipfile.ZipFile(BytesIO(merged), "r") as zf:
- assert zf.read("Metadata/plate_1.gcode") == b"GCODE-PLATE-1"
- assert zf.read("Metadata/plate_2.gcode") == b"GCODE-PLATE-2"
- assert zf.read("Metadata/plate_3.gcode") == b"GCODE-PLATE-3"
- # Per-plate thumbnails and json overlaid too.
- assert zf.read("Metadata/plate_2.png") == b"PLATE_PNG"
- assert zf.read("Metadata/plate_3_small.png") == b"SMALL"
- # Base 3MF's project_settings.config carried forward unchanged.
- assert zf.read("Metadata/project_settings.config") == p1_project(p1)
- def test_combined_slice_info_lists_every_plate(self):
- p1 = self._single_plate_3mf(1, b"G1", slice_info_block='<plate><metadata key="index" value="1"/></plate>')
- p2 = self._single_plate_3mf(2, b"G2", slice_info_block='<plate><metadata key="index" value="2"/></plate>')
- merged = merge_plate_3mfs([(1, p1), (2, p2)])
- with zipfile.ZipFile(BytesIO(merged), "r") as zf:
- info = zf.read("Metadata/slice_info.config").decode("utf-8")
- # Both plate blocks present.
- assert 'value="1"' in info
- assert 'value="2"' in info
- # Two <plate> blocks total (we don't include the source's stale
- # one from before slicing).
- assert info.count("<plate>") == 2
- def test_falls_back_to_source_thumbnails_when_sliced_outputs_lack_them(self):
- """BS CLI with --arrange generates fresh per-plate gcode but
- doesn't always write a fresh ``plate_N.png``. The merger's
- ``source_3mf_bytes`` fallback should fill the gap from the
- source 3MF's original per-plate render so the archive's per-
- plate previews aren't blank."""
- # Sliced outputs that lack plate_N.png entries entirely (only
- # gcode + json + md5 — the thumbnail slot is empty).
- def _no_thumb_3mf(plate_num: int) -> bytes:
- return _make_3mf(
- {
- "3D/3dmodel.model": b"<model/>",
- "Metadata/project_settings.config": b"{}",
- "Metadata/model_settings.config": b"<config/>",
- "Metadata/slice_info.config": (
- '<?xml version="1.0"?>\n<config>'
- f'<plate><metadata key="index" value="{plate_num}"/></plate>'
- "</config>"
- ).encode(),
- f"Metadata/plate_{plate_num}.gcode": f"G{plate_num}".encode(),
- }
- )
- source = _make_3mf(
- {
- "3D/3dmodel.model": b"<model/>",
- "Metadata/plate_1.png": b"SRC_PNG_1",
- "Metadata/plate_1_small.png": b"SRC_SMALL_1",
- "Metadata/plate_2.png": b"SRC_PNG_2",
- "Metadata/plate_2_small.png": b"SRC_SMALL_2",
- }
- )
- merged = merge_plate_3mfs(
- [(1, _no_thumb_3mf(1)), (2, _no_thumb_3mf(2))],
- source_3mf_bytes=source,
- )
- with zipfile.ZipFile(BytesIO(merged), "r") as zf:
- assert zf.read("Metadata/plate_1.png") == b"SRC_PNG_1"
- assert zf.read("Metadata/plate_1_small.png") == b"SRC_SMALL_1"
- assert zf.read("Metadata/plate_2.png") == b"SRC_PNG_2"
- assert zf.read("Metadata/plate_2_small.png") == b"SRC_SMALL_2"
- def test_source_fallback_does_not_overwrite_fresh_sliced_thumbnails(self):
- """If a sliced output DID write its own ``plate_N.png`` (same-class
- slice / older BS that renders even with arrange), keep it — the
- sliced render reflects the actual H2D layout; the source fallback
- only fills gaps."""
- p1 = self._single_plate_3mf(1, b"G1") # has plate_1.png = PLATE_PNG
- p2 = self._single_plate_3mf(2, b"G2") # has plate_2.png = PLATE_PNG
- source = _make_3mf(
- {
- "Metadata/plate_1.png": b"SRC_PNG_1",
- "Metadata/plate_2.png": b"SRC_PNG_2",
- }
- )
- merged = merge_plate_3mfs([(1, p1), (2, p2)], source_3mf_bytes=source)
- with zipfile.ZipFile(BytesIO(merged), "r") as zf:
- # Sliced output's thumbnails win.
- assert zf.read("Metadata/plate_1.png") == b"PLATE_PNG"
- assert zf.read("Metadata/plate_2.png") == b"PLATE_PNG"
- def test_unsorted_input_is_sorted_by_plate_number(self):
- p1 = self._single_plate_3mf(1, b"G1")
- p2 = self._single_plate_3mf(2, b"G2")
- # Pass them out of order; the merger should still place plate 2's
- # artifacts at plate_2.* and plate 1's at plate_1.*.
- merged = merge_plate_3mfs([(2, p2), (1, p1)])
- with zipfile.ZipFile(BytesIO(merged), "r") as zf:
- assert zf.read("Metadata/plate_1.gcode") == b"G1"
- assert zf.read("Metadata/plate_2.gcode") == b"G2"
- def p1_project(zip_bytes: bytes) -> bytes:
- """Helper for the merge test — pulls plate-1's project_settings.config out
- of a fixture so the test's assertion shows the actual reference value
- rather than hard-coding the literal."""
- with zipfile.ZipFile(BytesIO(zip_bytes), "r") as zf:
- return zf.read("Metadata/project_settings.config")
- class TestSubstituteUnusedPlateFilaments:
- """Slot 1 carries the used filament; unused-slot entries are
- overwritten with slot 1 so BambuStudio's filament-temp validator
- doesn't trip on heterogeneous loaded filaments that the plate's
- G-code never actually touches."""
- @staticmethod
- def _model_settings_xml(per_plate_extruders: list[tuple[int, list[int]]]) -> bytes:
- """Build a minimal model_settings.config mapping each plate to a set
- of extruder/slot numbers via per-object extruder metadata. Mirrors
- the schema ``extract_plate_extruder_set_from_3mf`` parses:
- - top-level ``<object id=N>`` with ``<metadata key="extruder" .../>``
- - per-plate ``<plate>`` listing the object ids it contains.
- ``per_plate_extruders`` is a list of (plate_id, [extruder_ids]).
- Object ids are auto-numbered globally so plates can reference them.
- """
- objects = []
- plates = []
- oid = 1
- for plate_id, exts in per_plate_extruders:
- plate_obj_refs = []
- for ext in exts:
- objects.append(f'<object id="{oid}"><metadata key="extruder" value="{ext}"/></object>')
- plate_obj_refs.append(
- f'<model_instance><metadata key="object_id" value="{oid}"/>'
- f'<metadata key="instance_id" value="0"/>'
- f'<metadata key="identify_id" value="{oid}"/></model_instance>'
- )
- oid += 1
- plates.append(
- f'<plate><metadata key="plater_id" value="{plate_id}"/>' + "".join(plate_obj_refs) + "</plate>"
- )
- xml = (
- '<?xml version="1.0" encoding="UTF-8"?>\n'
- "<config>\n" + "\n".join(objects) + "\n" + "\n".join(plates) + "\n" + "</config>"
- )
- return xml.encode("utf-8")
- def test_substitutes_unused_slot_with_slot_1(self):
- # Plate 1 uses slot 1 only; slots 2 and 3 are loaded but unused.
- zip_bytes = _make_3mf({"Metadata/model_settings.config": self._model_settings_xml([(1, [1])])})
- items = ["pla_basic.json", "abs_loaded_but_unused.json", "abs_again_unused.json"]
- result = substitute_unused_plate_filaments(zip_bytes, plate_id=1, items=items)
- assert result == ["pla_basic.json", "pla_basic.json", "pla_basic.json"]
- def test_no_substitution_when_all_used(self):
- # Multi-colour plate where every slot is used: leave the user's selections alone.
- zip_bytes = _make_3mf({"Metadata/model_settings.config": self._model_settings_xml([(1, [1, 2, 3])])})
- items = ["pla_white.json", "pla_red.json", "pla_blue.json"]
- result = substitute_unused_plate_filaments(zip_bytes, plate_id=1, items=items)
- assert result == ["pla_white.json", "pla_red.json", "pla_blue.json"]
- def test_no_op_when_plate_id_is_none(self):
- items = ["a.json", "b.json", "c.json"]
- result = substitute_unused_plate_filaments(b"any bytes", plate_id=None, items=items)
- assert result == items
- def test_no_op_when_single_filament(self):
- result = substitute_unused_plate_filaments(b"any bytes", plate_id=1, items=["only.json"])
- assert result == ["only.json"]
- result = substitute_unused_plate_filaments(b"any bytes", plate_id=1, items=[])
- assert result == []
- def test_no_op_when_source_not_zip(self):
- items = ["a.json", "b.json"]
- result = substitute_unused_plate_filaments(b"not a zip", plate_id=1, items=items)
- assert result == items
- def test_no_op_when_no_model_settings(self):
- # Empty parse result is treated as "every slot used" (fail-open default).
- zip_bytes = _make_3mf({"3D/3dmodel.model": b"<model/>"})
- items = ["a.json", "b.json", "c.json"]
- result = substitute_unused_plate_filaments(zip_bytes, plate_id=1, items=items)
- assert result == items
|