| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331 |
- """Pure helper functions for Spoolman spool mapping.
- No heavy dependencies — importable in unit tests without the full backend stack.
- """
- from __future__ import annotations
- import ipaddress
- import json
- import logging
- import math
- import re
- from typing import Any
- from urllib.parse import urlparse
- from typing_extensions import TypedDict
- from backend.app.api.routes._url_safety import CLOUD_METADATA_IPS, NUMERIC_IP_RE, unwrap_ipv4_mapped
- logger = logging.getLogger(__name__)
- class MappedSpoolFields(TypedDict):
- """Full shape of the dict returned by _map_spoolman_spool (InventorySpool-compatible)."""
- id: int
- material: str | None
- subtype: str | None
- brand: str | None
- color_name: str | None
- color_name_is_synthesized: bool
- rgba: str | None
- label_weight: int | None
- core_weight: int | None
- core_weight_catalog_id: None
- weight_used: float | None
- weight_locked: bool
- last_scale_weight: None
- last_weighed_at: None
- slicer_filament: None
- slicer_filament_name: str | None
- nozzle_temp_min: int | None
- nozzle_temp_max: None
- note: str | None
- added_full: None
- last_used: str | None
- encode_time: str | None
- tag_uid: str | None
- tray_uuid: str | None
- data_origin: str | None
- tag_type: str | None
- archived_at: str | None
- created_at: str | None # None when Spoolman spool has no registered timestamp
- updated_at: str | None
- cost_per_kg: float | None
- storage_location: str | None
- k_profiles: list[Any]
- class NormalizedVendorRef(TypedDict):
- """Vendor reference embedded in a NormalizedFilament."""
- id: int
- name: str
- class NormalizedFilament(TypedDict):
- """Normalised Spoolman filament dict returned by the /filaments catalog endpoint."""
- id: int
- name: str
- material: str | None
- color_hex: str | None
- color_name: str | None
- weight: int | None
- spool_weight: float | None
- vendor: NormalizedVendorRef | None
- def assert_safe_spoolman_url(url: str) -> None:
- """Raise ValueError if *url* should be blocked as an SSRF risk.
- Bambuddy is typically deployed on a home LAN alongside Spoolman, so
- loopback (127.0.0.1) and RFC-1918 private ranges (192.168.x.x, 10.x.x.x,
- 172.16-31.x) must be permitted — they are THE normal Spoolman topology.
- This guard therefore targets the genuinely dangerous cases only.
- Checks performed:
- - Scheme must be http or https (no file://, gopher://, dict://, etc.).
- - Numeric-encoded IP addresses in decimal (e.g. ``2130706433``) or hex
- (e.g. ``0x7f000001``) are rejected. Python's ``ipaddress`` module raises
- ``ValueError`` for these forms so they would otherwise bypass the
- explicit-IP block below, but libc (and browsers) resolve them as valid
- IPv4 addresses.
- - Cloud provider metadata endpoints (169.254.169.254, 100.100.100.200,
- fd00:ec2::254) are blocked — the classic SSRF credential-exfil target.
- - Multicast (224.0.0.0/4, ff00::/8) and unspecified (0.0.0.0, ::) addresses
- are blocked — pointless as a destination and suggests misuse.
- - IPv4-mapped IPv6 addresses (::ffff:x.x.x.x) are unwrapped so they cannot
- bypass the checks above.
- Hostname-based addresses ("localhost", "spoolman.lan", "internal.corp")
- are out of scope — DNS resolution is deliberately not performed here.
- """
- parsed = urlparse(url)
- if parsed.scheme.lower() not in ("http", "https"):
- raise ValueError("Spoolman URL must use http or https")
- hostname = (parsed.hostname or "").lower()
- # Reject decimal- and hex-encoded IPs (e.g. http://2130706433/ or
- # http://0x7f000001/). These slip past ipaddress.ip_address() but libc
- # (and browsers) parse them as IPv4 — an obvious bypass if not caught.
- if NUMERIC_IP_RE.match(hostname):
- raise ValueError("Spoolman URL must not use numeric-encoded IP addresses; use standard dotted-decimal notation")
- try:
- addr = ipaddress.ip_address(hostname)
- except ValueError:
- # Not a bare IP address — includes intentional cases such as "localhost" and
- # RFC-1918 hostnames ("spoolman.lan", "192.168.1.10" would be caught above as
- # a dotted-decimal IP; symbolic names resolve via DNS which is out of scope).
- # Running Spoolman on the same host or home LAN is the standard Bambuddy
- # topology, so loopback and private ranges are deliberately NOT blocked here.
- return
- # Unwrap IPv4-mapped IPv6 (::ffff:169.254.169.254 etc.) so attackers can't
- # encode a blocked IPv4 into an IPv6 literal to bypass the check.
- effective = unwrap_ipv4_mapped(addr)
- if effective in CLOUD_METADATA_IPS:
- raise ValueError("Spoolman URL must not point to a cloud metadata endpoint")
- if effective.is_multicast or effective.is_unspecified:
- raise ValueError("Spoolman URL must not point to a multicast or unspecified address")
- _COLOR_HEX_RE = re.compile(r"^[0-9A-Fa-f]{6}$")
- _TAG_HEX_RE = re.compile(r"^[0-9A-F]+$")
- def _safe_int(value: object, fallback: int) -> int:
- """Convert value to int, returning fallback for None/NaN/Inf/non-numeric."""
- try:
- f = float(value) # type: ignore[arg-type]
- if math.isfinite(f):
- return int(f)
- except (TypeError, ValueError):
- pass
- return fallback
- def _safe_float(value: object, fallback: float) -> float:
- """Convert value to float, returning fallback for None/NaN/Inf/non-numeric."""
- try:
- f = float(value) # type: ignore[arg-type]
- if math.isfinite(f):
- return f
- except (TypeError, ValueError):
- pass
- return fallback
- def _safe_optional_float(value: object) -> float | None:
- """Convert value to finite float, or None if missing/NaN/Infinite/non-numeric.
- Used for optional monetary fields (price) to prevent Infinity/NaN from
- reaching JSON serialisation, which raises ValueError with allow_nan=False.
- """
- if value is None:
- return None
- try:
- f = float(value) # type: ignore[arg-type]
- if math.isfinite(f):
- return f
- except (TypeError, ValueError):
- pass
- return None
- def _extract_extra_str(extra: dict, key: str) -> str:
- """Extract a JSON-encoded string from a Spoolman extra dict.
- Spoolman stores extra values as JSON-stringified text — a stored string
- "GFL05" appears as `'"GFL05"'` (six chars including the quotes). This
- unwraps that, returning the bare string. Returns "" for missing keys,
- non-strings, or invalid JSON.
- """
- raw = extra.get(key)
- if not isinstance(raw, str):
- return ""
- try:
- decoded = json.loads(raw)
- except (json.JSONDecodeError, ValueError):
- # Tolerate bare-string values written without JSON encoding.
- return raw
- return decoded if isinstance(decoded, str) else ""
- def _map_spoolman_spool(spool: dict) -> MappedSpoolFields:
- """Convert a raw Spoolman spool dict to the InventorySpool-compatible format.
- Fields not supported by Spoolman (k_profiles, slicer_filament, …) are
- returned as None / empty so the frontend can still render them without
- errors. The ``data_origin`` field is set to ``"spoolman"`` so UI code can
- distinguish these spools from local ones.
- """
- raw_id = spool.get("id")
- if raw_id is None:
- raise ValueError("Spoolman spool is missing required 'id' field")
- try:
- spool_id: int = int(raw_id)
- except (TypeError, ValueError):
- raise ValueError(f"Spoolman spool 'id' is not a valid integer: {raw_id!r}")
- if spool_id <= 0:
- raise ValueError(f"Spoolman spool 'id' must be a positive integer, got {spool_id}")
- filament: dict = spool.get("filament") or {}
- if not filament:
- logger.warning(
- "Spoolman spool %s has no filament data — all filament fields will use defaults",
- spool_id,
- )
- vendor: dict = filament.get("vendor") or {}
- extra: dict = spool.get("extra") or {}
- # RFID tag stored as JSON-encoded string in Spoolman extra.tag.
- # 32-char hex → Bambu Lab tray UUID; 8–30-char hex → NFC tag UID.
- # Accepting the full realistic UID range (4-byte = 8 chars, 7-byte = 14 chars,
- # 10-byte = 20 chars) avoids silently dropping valid SpoolBuddy-written tags.
- raw_tag: str = (extra.get("tag") or "").strip('"').upper()
- _raw_is_hex = bool(_TAG_HEX_RE.match(raw_tag))
- tag_uid = raw_tag if _raw_is_hex and 8 <= len(raw_tag) <= 30 else None
- tray_uuid = raw_tag if _raw_is_hex and len(raw_tag) == 32 else None
- # Subtype = filament name with material prefix stripped
- material: str = (filament.get("material") or "").strip()
- filament_name: str = (filament.get("name") or "").strip()
- if material and filament_name.upper().startswith(material.upper()):
- subtype: str | None = filament_name[len(material) :].strip() or None
- else:
- subtype = filament_name or None
- # Colour: validate as 6-char hex; fall back to neutral grey for invalid values
- raw_color = (filament.get("color_hex") or "").upper().removeprefix("#")
- color_hex: str = raw_color if _COLOR_HEX_RE.match(raw_color) else "808080"
- rgba: str = color_hex + "FF"
- label_weight: int = _safe_int(filament.get("weight"), 1000)
- used_weight: float = _safe_float(spool.get("used_weight"), 0.0)
- # Archived state – Spoolman uses a boolean ``archived`` field
- archived: bool = spool.get("archived", False)
- archived_at: str | None = None
- if archived:
- archived_at = spool.get("last_used") or spool.get("registered") or None
- created_at: str | None = spool.get("registered") or None
- # Spoolman has no `color_name` field on Filament — confirmed against the
- # FilamentUpdateParameters schema in 0.23.1: name/vendor_id/material/price/
- # density/diameter/weight/spool_weight/article_number/comment/extruder_temp/
- # bed_temp/color_hex/multi_color_hexes/multi_color_direction/external_id/
- # extra, no color_name (#1357). The previous attempt (b8e350c3) was
- # PATCHing a key Spoolman silently discards, which is why color_name
- # never actually persisted from the user's edits.
- #
- # We persist it ourselves under spool.extra.bambu_color_name (JSON-encoded
- # string, same pattern as bambu_slicer_filament). Read order:
- # 1. spool.extra.bambu_color_name (the canonical store)
- # 2. filament.color_name (forward-compat — picks up the value if a
- # future Spoolman release adds the field, or if an admin populated
- # it via a custom extra-field they registered themselves)
- # 3. subtype (synth fallback so the inventory list isn't a sea of
- # "Unknown color" entries on installs with neither field set)
- #
- # color_name_is_synthesized = True only when we fell back to subtype.
- # The edit form uses it to leave the input blank, so the user doesn't
- # round-trip the synth value back as if they had set it.
- extra_color_name = _extract_extra_str(extra, "bambu_color_name") or None
- stored_color_name = extra_color_name or (filament.get("color_name") or None)
- color_name: str | None = stored_color_name or subtype or None
- color_name_is_synthesized: bool = stored_color_name is None and color_name is not None
- nozzle_temp_raw = filament.get("settings_extruder_temp")
- nozzle_temp_min: int | None = _safe_int(nozzle_temp_raw, 0) or None
- return {
- "id": spool_id,
- "material": material,
- "subtype": subtype,
- "color_name": color_name,
- "color_name_is_synthesized": color_name_is_synthesized,
- "rgba": rgba,
- "brand": vendor.get("name") or None,
- "label_weight": label_weight,
- "core_weight": _safe_int(
- spool.get("spool_weight") if spool.get("spool_weight") is not None else filament.get("spool_weight"), 250
- ),
- "core_weight_catalog_id": None,
- "weight_used": used_weight,
- "weight_locked": False,
- "last_scale_weight": None,
- "last_weighed_at": None,
- # BambuStudio slicer preset — Spoolman has no native field, so the
- # update endpoint persists these under bambu_slicer_filament[_name]
- # in the spool's extra dict. Values are JSON-encoded strings; an
- # empty string ("") means cleared. Falls back to Spoolman's
- # filament_name for slicer_filament_name when nothing is stored.
- "slicer_filament": (_extract_extra_str(extra, "bambu_slicer_filament") or None),
- "slicer_filament_name": (_extract_extra_str(extra, "bambu_slicer_filament_name") or (filament_name or None)),
- "nozzle_temp_min": nozzle_temp_min,
- "nozzle_temp_max": None,
- "note": spool.get("comment") or None,
- "added_full": None,
- "last_used": spool.get("last_used"),
- # encode_time semantics differ: local records NFC write time; Spoolman first_used
- # records first print use — different events; using first_used as best available proxy.
- "encode_time": spool.get("first_used"),
- "tag_uid": tag_uid,
- "tray_uuid": tray_uuid,
- "data_origin": "spoolman",
- "tag_type": "spoolman",
- "archived_at": archived_at,
- "created_at": created_at,
- # Spoolman has no updated_at field; use registered timestamp as best available proxy
- "updated_at": created_at,
- "cost_per_kg": _safe_optional_float(spool.get("price")),
- "storage_location": spool.get("location") or None,
- "k_profiles": [],
- }
|