slicer_presets.py 2.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. """Pydantic schemas for the unified slicer-presets endpoint.
  2. The SliceModal pulls printer/process/filament options from three sources, in
  3. priority order: cloud (the user's Bambu Cloud account), local (DB-backed
  4. imported profiles), and standard (slicer-bundled stock profiles). The endpoint
  5. returns all three lists with name-based dedup applied so each preset appears
  6. exactly once across the response.
  7. """
  8. from typing import Literal
  9. from pydantic import BaseModel
  10. CloudStatus = Literal["ok", "not_authenticated", "expired", "unreachable"]
  11. class UnifiedPreset(BaseModel):
  12. """A single printer/process/filament preset with its source.
  13. The ``id`` shape varies by source:
  14. - cloud → Bambu Cloud setting_id (e.g. ``"PFUS9ac902733670a9"``)
  15. - local → stringified DB row id from ``local_presets``
  16. - standard → preset name as written in the bundled JSON (the slicer
  17. resolves bundled profiles by name during inheritance walk)
  18. The frontend treats ``id`` as opaque; the slice dispatch path uses
  19. ``(source, id)`` to fetch / pass the preset content to the sidecar.
  20. ``filament_type`` and ``filament_colour`` are populated for the filament
  21. slot only — they let the SliceModal pre-pick a preset per plate slot in
  22. the multi-color flow by matching against the source 3MF's per-slot type
  23. and color. Populated when the underlying preset JSON exposes them; left
  24. as ``None`` on bundled profiles where colour is a runtime spool attribute.
  25. """
  26. id: str
  27. name: str
  28. source: Literal["cloud", "local", "standard"]
  29. filament_type: str | None = None
  30. filament_colour: str | None = None
  31. class UnifiedPresetsBySlot(BaseModel):
  32. """Three slots in the order Bambu Studio / OrcaSlicer use."""
  33. printer: list[UnifiedPreset] = []
  34. process: list[UnifiedPreset] = []
  35. filament: list[UnifiedPreset] = []
  36. class UnifiedPresetsResponse(BaseModel):
  37. """Each tier carries only the names that didn't appear in a higher tier.
  38. Cloud is the highest priority (user's personal customisations win), then
  39. the local imports the user explicitly curated, then the slicer's stock
  40. fallback. A name that appears in cloud is filtered out of local and
  41. standard; a name that appears in local is filtered out of standard.
  42. ``cloud_status`` lets the frontend show a banner explaining why the cloud
  43. tier is empty when the user expected to see it (signed out / token
  44. expired / network down).
  45. """
  46. cloud: UnifiedPresetsBySlot = UnifiedPresetsBySlot()
  47. local: UnifiedPresetsBySlot = UnifiedPresetsBySlot()
  48. standard: UnifiedPresetsBySlot = UnifiedPresetsBySlot()
  49. cloud_status: CloudStatus = "ok"