| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027 |
- """
- Bambu Lab Cloud API Routes
- Handles authentication and profile management with Bambu Cloud.
- """
- import json
- import logging
- from pathlib import Path
- from typing import Literal
- from fastapi import APIRouter, Body, Depends, HTTPException
- from sqlalchemy import select
- from sqlalchemy.ext.asyncio import AsyncSession
- from backend.app.core.auth import RequirePermissionIfAuthEnabled
- from backend.app.core.database import get_db
- from backend.app.core.permissions import Permission
- from backend.app.models.settings import Settings
- from backend.app.models.user import User
- from backend.app.schemas.cloud import (
- CloudAuthStatus,
- CloudDevice,
- CloudLoginRequest,
- CloudLoginResponse,
- CloudTokenRequest,
- CloudVerifyRequest,
- FirmwareUpdateInfo,
- FirmwareUpdatesResponse,
- SlicerSetting,
- SlicerSettingCreate,
- SlicerSettingDeleteResponse,
- SlicerSettingsResponse,
- SlicerSettingUpdate,
- )
- from backend.app.services.bambu_cloud import (
- BambuCloudAuthError,
- BambuCloudError,
- BambuCloudService,
- )
- from backend.app.utils.filament_ids import filament_id_to_setting_id
- logger = logging.getLogger(__name__)
- router = APIRouter(prefix="/cloud", tags=["cloud"])
- # Keys for storing cloud credentials in settings
- CLOUD_TOKEN_KEY = "bambu_cloud_token"
- CLOUD_EMAIL_KEY = "bambu_cloud_email"
- CLOUD_REGION_KEY = "bambu_cloud_region"
- def _normalise_region(region: str | None) -> str:
- """Treat NULL/empty as 'global' for legacy rows that predate the region column."""
- return region if region in ("global", "china") else "global"
- async def get_stored_token(db: AsyncSession, user: User | None = None) -> tuple[str | None, str | None, str]:
- """Get stored cloud token, email, and region.
- When a user is provided (auth enabled), returns that user's per-user credentials.
- When user is None (auth disabled), falls back to global Settings table.
- Region defaults to ``"global"`` when unset (including for rows that predate
- the ``cloud_region`` column).
- """
- if user is not None:
- return user.cloud_token, user.cloud_email, _normalise_region(user.cloud_region)
- # Fallback: global storage (auth disabled)
- result = await db.execute(
- select(Settings).where(Settings.key.in_([CLOUD_TOKEN_KEY, CLOUD_EMAIL_KEY, CLOUD_REGION_KEY]))
- )
- settings = {s.key: s.value for s in result.scalars().all()}
- return (
- settings.get(CLOUD_TOKEN_KEY),
- settings.get(CLOUD_EMAIL_KEY),
- _normalise_region(settings.get(CLOUD_REGION_KEY)),
- )
- async def store_token(db: AsyncSession, token: str, email: str, region: str, user: User | None = None) -> None:
- """Store cloud token, email, and region.
- When a user is provided (auth enabled), stores on the user record.
- When user is None (auth disabled), stores in global Settings table.
- """
- region = _normalise_region(region)
- if user is not None:
- # User object is from the auth dependency's session (detached),
- # so use a direct UPDATE via the route's db session.
- from sqlalchemy import update
- await db.execute(
- update(User).where(User.id == user.id).values(cloud_token=token, cloud_email=email, cloud_region=region)
- )
- await db.commit()
- return
- # Fallback: global storage (auth disabled)
- for key, value in [(CLOUD_TOKEN_KEY, token), (CLOUD_EMAIL_KEY, email), (CLOUD_REGION_KEY, region)]:
- result = await db.execute(select(Settings).where(Settings.key == key))
- setting = result.scalar_one_or_none()
- if setting:
- setting.value = value
- else:
- db.add(Settings(key=key, value=value))
- await db.commit()
- async def clear_token(db: AsyncSession, user: User | None = None) -> None:
- """Clear stored cloud token, email, and region.
- When a user is provided (auth enabled), clears that user's credentials.
- When user is None (auth disabled), clears from global Settings table.
- """
- if user is not None:
- from sqlalchemy import update
- await db.execute(
- update(User).where(User.id == user.id).values(cloud_token=None, cloud_email=None, cloud_region=None)
- )
- await db.commit()
- return
- # Fallback: global storage (auth disabled)
- result = await db.execute(
- select(Settings).where(Settings.key.in_([CLOUD_TOKEN_KEY, CLOUD_EMAIL_KEY, CLOUD_REGION_KEY]))
- )
- for setting in result.scalars().all():
- await db.delete(setting)
- await db.commit()
- async def build_authenticated_cloud(db: AsyncSession, user: User | None) -> BambuCloudService | None:
- """Build a per-request cloud service seeded with the caller's stored token + region.
- Returns ``None`` when no token is stored, so callers can 401 without constructing
- (and then closing) a useless client. Caller is responsible for ``await cloud.close()``.
- """
- token, _email, region = await get_stored_token(db, user)
- if not token:
- return None
- cloud = BambuCloudService(region=region)
- cloud.set_token(token)
- return cloud
- @router.get("/status", response_model=CloudAuthStatus)
- async def get_auth_status(
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """Get current cloud authentication status.
- Reads the stored credentials in one DB round-trip (we used to call
- ``get_stored_token`` twice — once here and once inside
- ``build_authenticated_cloud``). ``region`` is exposed so the frontend can
- show "Connected (China)" after a reload without relying on local state.
- """
- token, email, region = await get_stored_token(db, current_user)
- if not token:
- return CloudAuthStatus(is_authenticated=False, email=None, region=None)
- cloud = BambuCloudService(region=region)
- cloud.set_token(token)
- try:
- authenticated = cloud.is_authenticated
- return CloudAuthStatus(
- is_authenticated=authenticated,
- email=email if authenticated else None,
- region=region if authenticated else None,
- )
- finally:
- await cloud.close()
- @router.post("/login", response_model=CloudLoginResponse)
- async def login(
- request: CloudLoginRequest,
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Initiate login to Bambu Cloud.
- This will trigger either:
- - Email verification: A code is sent to the user's email
- - TOTP verification: User enters code from their authenticator app
- After receiving/generating the code, call /cloud/verify to complete the login.
- For TOTP, include the tfa_key from this response in the verify request.
- """
- cloud = BambuCloudService(region=request.region)
- try:
- result = await cloud.login_request(request.email, request.password)
- if result.get("success") and cloud.access_token:
- # Direct login succeeded (rare)
- await store_token(db, cloud.access_token, request.email, request.region, current_user)
- return CloudLoginResponse(
- success=result.get("success", False),
- needs_verification=result.get("needs_verification", False),
- message=result.get("message", "Unknown error"),
- verification_type=result.get("verification_type"),
- tfa_key=result.get("tfa_key"),
- )
- except BambuCloudAuthError as e:
- raise HTTPException(status_code=401, detail=str(e))
- except BambuCloudError as e:
- raise HTTPException(status_code=500, detail=str(e))
- finally:
- await cloud.close()
- @router.post("/verify", response_model=CloudLoginResponse)
- async def verify_code(
- request: CloudVerifyRequest,
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Complete login with verification code (email or TOTP).
- For email verification:
- - After calling /cloud/login, the user receives an email with a 6-digit code
- - Submit the code with email address
- For TOTP verification:
- - The user enters the 6-digit code from their authenticator app
- - Include the tfa_key from the /cloud/login response
- ``request.region`` must match the region used in /cloud/login so that the
- TOTP call hits the correct TFA endpoint (bambulab.com vs bambulab.cn).
- """
- cloud = BambuCloudService(region=request.region)
- try:
- # Use TOTP verification if tfa_key is provided
- if request.tfa_key:
- result = await cloud.verify_totp(request.tfa_key, request.code)
- else:
- result = await cloud.verify_code(request.email, request.code)
- if result.get("success") and cloud.access_token:
- await store_token(db, cloud.access_token, request.email, request.region, current_user)
- return CloudLoginResponse(
- success=result.get("success", False),
- needs_verification=False,
- message=result.get("message", "Unknown error"),
- )
- except BambuCloudAuthError as e:
- raise HTTPException(status_code=401, detail=str(e))
- except BambuCloudError as e:
- raise HTTPException(status_code=500, detail=str(e))
- finally:
- await cloud.close()
- @router.post("/token", response_model=CloudAuthStatus)
- async def set_token(
- request: CloudTokenRequest,
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Set access token directly.
- For users who already have a token (e.g., from Bambu Studio). The
- selected ``region`` is persisted alongside the token so every subsequent
- request hits the right Bambu API endpoint, including after a restart.
- """
- cloud = BambuCloudService(region=request.region)
- cloud.set_token(request.access_token)
- try:
- # Verify token works by trying to get profile
- await cloud.get_user_profile()
- await store_token(db, request.access_token, "token-auth", request.region, current_user)
- return CloudAuthStatus(is_authenticated=True, email="token-auth")
- except BambuCloudError:
- raise HTTPException(status_code=401, detail="Invalid token")
- finally:
- await cloud.close()
- @router.post("/logout")
- async def logout(
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """Log out of Bambu Cloud."""
- await clear_token(db, current_user)
- return {"success": True}
- @router.get("/settings", response_model=SlicerSettingsResponse)
- async def get_slicer_settings(
- version: str = "02.04.00.70",
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Get all slicer settings (filament, printer, process presets).
- Requires authentication.
- """
- cloud = await build_authenticated_cloud(db, current_user)
- if cloud is None or not cloud.is_authenticated:
- raise HTTPException(status_code=401, detail="Not authenticated")
- try:
- data = await cloud.get_slicer_settings(version)
- result = SlicerSettingsResponse()
- # Map API keys to our types (API uses 'print' for process presets)
- type_mapping = {
- "filament": "filament",
- "printer": "printer",
- "print": "process", # API calls it 'print', we call it 'process'
- }
- for api_key, our_type in type_mapping.items():
- type_data = data.get(api_key, {})
- private_settings = type_data.get("private", [])
- public_settings = type_data.get("public", [])
- parsed = []
- # Private (custom) presets first
- for s in private_settings:
- parsed.append(
- SlicerSetting(
- setting_id=s.get("setting_id", s.get("id", "")),
- name=s.get("name", "Unknown"),
- type=our_type,
- version=s.get("version"),
- user_id=s.get("user_id"),
- updated_time=s.get("updated_time"),
- is_custom=True,
- )
- )
- # Public (default) presets
- for s in public_settings:
- parsed.append(
- SlicerSetting(
- setting_id=s.get("setting_id", s.get("id", "")),
- name=s.get("name", "Unknown"),
- type=our_type,
- version=s.get("version"),
- user_id=s.get("user_id"),
- updated_time=s.get("updated_time"),
- is_custom=False,
- )
- )
- setattr(result, our_type, parsed)
- return result
- except BambuCloudAuthError:
- await clear_token(db, current_user)
- raise HTTPException(status_code=401, detail="Authentication expired")
- except BambuCloudError as e:
- raise HTTPException(status_code=500, detail=str(e))
- finally:
- await cloud.close()
- @router.get("/settings/{setting_id}")
- async def get_setting_detail(
- setting_id: str,
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Get detailed information for a specific setting/preset.
- Returns the full preset configuration.
- """
- cloud = await build_authenticated_cloud(db, current_user)
- if cloud is None or not cloud.is_authenticated:
- raise HTTPException(status_code=401, detail="Not authenticated")
- try:
- data = await cloud.get_setting_detail(setting_id)
- return data
- except BambuCloudAuthError:
- await clear_token(db, current_user)
- raise HTTPException(status_code=401, detail="Authentication expired")
- except BambuCloudError as e:
- raise HTTPException(status_code=500, detail=str(e))
- finally:
- await cloud.close()
- @router.get("/filaments", response_model=list[SlicerSetting])
- async def get_filament_presets(
- version: str = "02.04.00.70",
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.FILAMENTS_READ),
- ):
- """
- Get just filament presets (convenience endpoint).
- Returns all filament presets with custom presets first.
- Uses the same cache as get_slicer_settings.
- """
- settings = await get_slicer_settings(version=version, db=db, current_user=current_user)
- return settings.filament
- # Cache for filament preset info (setting_id -> {name, k})
- _filament_cache: dict[str, dict] = {}
- _filament_cache_time: float = 0
- FILAMENT_CACHE_TTL = 300 # 5 minutes
- # Built-in filament ID → name mapping (fallback when cloud API and local profiles
- # don't have the entry). Based on Bambu Lab's known filament catalogue.
- _BUILTIN_FILAMENT_NAMES: dict[str, str] = {
- "GFA00": "Bambu PLA Basic",
- "GFA01": "Bambu PLA Matte",
- "GFA02": "Bambu PLA Metal",
- "GFA05": "Bambu PLA Silk",
- "GFA06": "Bambu PLA Silk+",
- "GFA07": "Bambu PLA Marble",
- "GFA08": "Bambu PLA Sparkle",
- "GFA09": "Bambu PLA Tough",
- "GFA11": "Bambu PLA Aero",
- "GFA12": "Bambu PLA Glow",
- "GFA13": "Bambu PLA Dynamic",
- "GFA15": "Bambu PLA Galaxy",
- "GFA16": "Bambu PLA Wood",
- "GFA50": "Bambu PLA-CF",
- "GFB00": "Bambu ABS",
- "GFB01": "Bambu ASA",
- "GFB02": "Bambu ASA-Aero",
- "GFB50": "Bambu ABS-GF",
- "GFB51": "Bambu ASA-CF",
- "GFB60": "PolyLite ABS",
- "GFB61": "PolyLite ASA",
- "GFB98": "Generic ASA",
- "GFB99": "Generic ABS",
- "GFC00": "Bambu PC",
- "GFC01": "Bambu PC FR",
- "GFC99": "Generic PC",
- "GFG00": "Bambu PETG Basic",
- "GFG01": "Bambu PETG Translucent",
- "GFG02": "Bambu PETG HF",
- "GFG50": "Bambu PETG-CF",
- "GFG60": "PolyLite PETG",
- "GFG96": "Generic PETG HF",
- "GFG97": "Generic PCTG",
- "GFG98": "Generic PETG-CF",
- "GFG99": "Generic PETG",
- "GFL00": "PolyLite PLA",
- "GFL01": "PolyTerra PLA",
- "GFL03": "eSUN PLA+",
- "GFL04": "Overture PLA",
- "GFL05": "Overture Matte PLA",
- "GFL06": "Fiberon PETG-ESD",
- "GFL50": "Fiberon PA6-CF",
- "GFL51": "Fiberon PA6-GF",
- "GFL52": "Fiberon PA12-CF",
- "GFL53": "Fiberon PA612-CF",
- "GFL54": "Fiberon PET-CF",
- "GFL55": "Fiberon PETG-rCF",
- "GFL95": "Generic PLA High Speed",
- "GFL96": "Generic PLA Silk",
- "GFL98": "Generic PLA-CF",
- "GFL99": "Generic PLA",
- "GFN03": "Bambu PA-CF",
- "GFN04": "Bambu PAHT-CF",
- "GFN05": "Bambu PA6-CF",
- "GFN06": "Bambu PPA-CF",
- "GFN08": "Bambu PA6-GF",
- "GFN96": "Generic PPA-GF",
- "GFN97": "Generic PPA-CF",
- "GFN98": "Generic PA-CF",
- "GFN99": "Generic PA",
- "GFP95": "Generic PP-GF",
- "GFP96": "Generic PP-CF",
- "GFP97": "Generic PP",
- "GFP98": "Generic PE-CF",
- "GFP99": "Generic PE",
- "GFR98": "Generic PHA",
- "GFR99": "Generic EVA",
- "GFS00": "Bambu Support W",
- "GFS01": "Bambu Support G",
- "GFS02": "Bambu Support For PLA",
- "GFS03": "Bambu Support For PA/PET",
- "GFS04": "Bambu PVA",
- "GFS05": "Bambu Support For PLA/PETG",
- "GFS06": "Bambu Support for ABS",
- "GFS97": "Generic BVOH",
- "GFS98": "Generic HIPS",
- "GFS99": "Generic PVA",
- "GFT01": "Bambu PET-CF",
- "GFT02": "Bambu PPS-CF",
- "GFT97": "Generic PPS",
- "GFT98": "Generic PPS-CF",
- "GFU00": "Bambu TPU 95A HF",
- "GFU01": "Bambu TPU 95A",
- "GFU02": "Bambu TPU for AMS",
- "GFU98": "Generic TPU for AMS",
- "GFU99": "Generic TPU",
- }
- async def _enrich_from_local_presets(
- unresolved_ids: list[str],
- result: dict,
- db: AsyncSession,
- ) -> dict:
- """Fall back to local profiles for filament IDs not resolved by cloud.
- Matches by checking the setting_id field inside the local preset's
- resolved JSON blob (stored in the 'setting' column).
- """
- from sqlalchemy import text
- from backend.app.models.local_preset import LocalPreset
- # Build lookup: converted setting_id -> original filament_id
- id_map: dict[str, str] = {}
- for fid in unresolved_ids:
- converted = _filament_id_to_setting_id(fid)
- id_map[converted] = fid
- # Also map the original in case the JSON uses that form
- id_map[fid] = fid
- try:
- # Query filament presets that have a setting_id matching any of our IDs
- from backend.app.core.db_dialect import is_sqlite
- if is_sqlite():
- json_filter = text("json_extract(setting, '$.setting_id') IS NOT NULL")
- else:
- json_filter = text("(setting::jsonb->>'setting_id') IS NOT NULL")
- candidates = await db.execute(
- select(LocalPreset).where(
- LocalPreset.preset_type == "filament",
- json_filter,
- )
- )
- for preset in candidates.scalars().all():
- try:
- setting_data = json.loads(preset.setting) if isinstance(preset.setting, str) else preset.setting
- preset_setting_id = setting_data.get("setting_id", "")
- if preset_setting_id in id_map:
- original_id = id_map[preset_setting_id]
- info = {"name": preset.name, "k": None}
- # Try to extract K value from the local preset
- pa = setting_data.get("pressure_advance")
- if pa is not None:
- try:
- k_val = float(pa[0]) if isinstance(pa, list) else float(pa)
- info["k"] = k_val
- except (ValueError, TypeError, IndexError):
- pass
- _filament_cache[original_id] = info
- result[original_id] = info
- except Exception:
- continue
- except Exception as e:
- logger.warning("Failed to search local presets for filament info: %s", e)
- # Phase 4: Fall back to built-in filament name table for any still without a name
- for fid in unresolved_ids:
- if fid not in result or not result[fid].get("name"):
- name = _BUILTIN_FILAMENT_NAMES.get(fid, "")
- if name:
- # Preserve K value from earlier phases if available
- existing_k = result.get(fid, {}).get("k")
- info = {"name": name, "k": existing_k}
- _filament_cache[fid] = info
- result[fid] = info
- # Fill remaining unresolved with empty entries
- for fid in unresolved_ids:
- if fid not in result:
- _filament_cache[fid] = {"name": "", "k": None}
- result[fid] = {"name": "", "k": None}
- return result
- # _filament_id_to_setting_id is now imported from backend.app.utils.filament_ids
- _filament_id_to_setting_id = filament_id_to_setting_id
- @router.post("/filament-info")
- async def get_filament_info(
- setting_ids: list[str] = Body(...),
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.FILAMENTS_READ),
- ):
- """
- Get filament preset info (name and K value) for multiple setting IDs.
- Used to enrich AMS tray and nozzle rack tooltips with preset data.
- Lookup order: cache → cloud → local profiles → built-in table → empty fallback.
- """
- import time
- logger.info("get_filament_info called with %s IDs: %s", len(setting_ids), setting_ids)
- global _filament_cache, _filament_cache_time
- # Clear stale cache
- if time.time() - _filament_cache_time > FILAMENT_CACHE_TTL:
- _filament_cache = {}
- _filament_cache_time = time.time()
- result = {}
- unresolved_ids: list[str] = []
- # Phase 1: Check cache
- for setting_id in setting_ids:
- if not setting_id:
- continue
- if setting_id in _filament_cache:
- result[setting_id] = _filament_cache[setting_id]
- else:
- unresolved_ids.append(setting_id)
- # Phase 2: Try cloud for uncached IDs
- if unresolved_ids:
- cloud = await build_authenticated_cloud(db, current_user)
- if cloud is not None and cloud.is_authenticated:
- try:
- still_unresolved: list[str] = []
- for setting_id in unresolved_ids:
- try:
- api_setting_id = _filament_id_to_setting_id(setting_id)
- data = await cloud.get_setting_detail(api_setting_id)
- setting = data.get("setting", {})
- name = data.get("name", "")
- k_value = setting.get("pressure_advance")
- if k_value is not None:
- try:
- k_value = float(k_value)
- except (ValueError, TypeError):
- k_value = None
- info = {"name": name, "k": k_value}
- _filament_cache[setting_id] = info
- result[setting_id] = info
- if not name:
- still_unresolved.append(setting_id)
- except Exception as e:
- logger.warning(
- f"Failed to get cloud preset {setting_id} "
- f"(API ID: {_filament_id_to_setting_id(setting_id)}): {e}"
- )
- still_unresolved.append(setting_id)
- unresolved_ids = still_unresolved
- finally:
- await cloud.close()
- elif cloud is not None:
- await cloud.close()
- # Phase 3: Try local profiles for any IDs still without a name
- if unresolved_ids:
- result = await _enrich_from_local_presets(unresolved_ids, result, db)
- return result
- @router.get("/devices", response_model=list[CloudDevice])
- async def get_devices(
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.PRINTERS_READ),
- ):
- """
- Get list of bound printer devices.
- Returns printers registered to the user's Bambu account.
- """
- cloud = await build_authenticated_cloud(db, current_user)
- if cloud is None or not cloud.is_authenticated:
- raise HTTPException(status_code=401, detail="Not authenticated")
- try:
- data = await cloud.get_devices()
- devices = data.get("devices", [])
- return [
- CloudDevice(
- dev_id=d.get("dev_id", ""),
- name=d.get("name", "Unknown"),
- dev_model_name=d.get("dev_model_name"),
- dev_product_name=d.get("dev_product_name"),
- online=d.get("online", False),
- )
- for d in devices
- ]
- except BambuCloudAuthError:
- await clear_token(db, current_user)
- raise HTTPException(status_code=401, detail="Authentication expired")
- except BambuCloudError as e:
- raise HTTPException(status_code=500, detail=str(e))
- finally:
- await cloud.close()
- @router.get("/firmware-updates", response_model=FirmwareUpdatesResponse)
- async def get_firmware_updates(
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.FIRMWARE_READ),
- ):
- """
- Check for firmware updates for all bound devices.
- Returns firmware version info for each device including:
- - Current installed version
- - Latest available version
- - Whether an update is available
- - Release notes for the latest version
- Requires cloud authentication.
- """
- cloud = await build_authenticated_cloud(db, current_user)
- if cloud is None or not cloud.is_authenticated:
- raise HTTPException(status_code=401, detail="Not authenticated")
- try:
- # First get list of bound devices
- devices_data = await cloud.get_devices()
- devices = devices_data.get("devices", [])
- updates = []
- updates_available = 0
- # Check firmware for each device
- for device in devices:
- device_id = device.get("dev_id", "")
- device_name = device.get("name", "Unknown")
- try:
- firmware_info = await cloud.get_firmware_version(device_id)
- update_available = firmware_info.get("update_available", False)
- if update_available:
- updates_available += 1
- updates.append(
- FirmwareUpdateInfo(
- device_id=device_id,
- device_name=device_name,
- current_version=firmware_info.get("current_version"),
- latest_version=firmware_info.get("latest_version"),
- update_available=update_available,
- release_notes=firmware_info.get("release_notes"),
- )
- )
- except BambuCloudError as e:
- logger.warning("Failed to get firmware info for %s: %s", device_name, e)
- # Still include device but with unknown firmware status
- updates.append(
- FirmwareUpdateInfo(
- device_id=device_id,
- device_name=device_name,
- current_version=None,
- latest_version=None,
- update_available=False,
- release_notes=None,
- )
- )
- return FirmwareUpdatesResponse(updates=updates, updates_available=updates_available)
- except BambuCloudAuthError:
- await clear_token(db, current_user)
- raise HTTPException(status_code=401, detail="Authentication expired")
- except BambuCloudError as e:
- raise HTTPException(status_code=500, detail=str(e))
- finally:
- await cloud.close()
- @router.post("/settings")
- async def create_setting(
- request: SlicerSettingCreate,
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Create a new slicer preset/setting.
- Creates a new preset on Bambu Cloud. The preset inherits from a base preset
- and only stores the delta (modified values).
- Type should be: 'filament', 'print', or 'printer'
- """
- cloud = await build_authenticated_cloud(db, current_user)
- if cloud is None or not cloud.is_authenticated:
- raise HTTPException(status_code=401, detail="Not authenticated")
- try:
- data = await cloud.create_setting(
- preset_type=request.type,
- name=request.name,
- base_id=request.base_id,
- setting=request.setting,
- version=request.version,
- )
- return data
- except BambuCloudAuthError:
- await clear_token(db, current_user)
- raise HTTPException(status_code=401, detail="Authentication expired")
- except BambuCloudError as e:
- raise HTTPException(status_code=500, detail=str(e))
- finally:
- await cloud.close()
- @router.put("/settings/{setting_id}")
- async def update_setting(
- setting_id: str,
- request: SlicerSettingUpdate,
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Update an existing slicer preset/setting.
- Updates the preset's name and/or settings on Bambu Cloud.
- """
- cloud = await build_authenticated_cloud(db, current_user)
- if cloud is None or not cloud.is_authenticated:
- raise HTTPException(status_code=401, detail="Not authenticated")
- try:
- data = await cloud.update_setting(
- setting_id=setting_id,
- name=request.name,
- setting=request.setting,
- )
- return data
- except BambuCloudAuthError:
- await clear_token(db, current_user)
- raise HTTPException(status_code=401, detail="Authentication expired")
- except BambuCloudError as e:
- raise HTTPException(status_code=500, detail=str(e))
- finally:
- await cloud.close()
- @router.delete("/settings/{setting_id}", response_model=SlicerSettingDeleteResponse)
- async def delete_setting(
- setting_id: str,
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Delete a slicer preset/setting.
- Removes the preset from Bambu Cloud. This cannot be undone.
- """
- cloud = await build_authenticated_cloud(db, current_user)
- if cloud is None or not cloud.is_authenticated:
- raise HTTPException(status_code=401, detail="Not authenticated")
- try:
- result = await cloud.delete_setting(setting_id)
- return SlicerSettingDeleteResponse(
- success=result.get("success", True),
- message=result.get("message", "Setting deleted"),
- )
- except BambuCloudAuthError:
- await clear_token(db, current_user)
- raise HTTPException(status_code=401, detail="Authentication expired")
- except BambuCloudError as e:
- raise HTTPException(status_code=500, detail=str(e))
- finally:
- await cloud.close()
- # Path to field definition files
- FIELDS_DATA_DIR = Path(__file__).parent.parent.parent / "data"
- # Cache for field definitions (loaded once)
- _fields_cache: dict[str, dict] = {}
- def _load_fields(preset_type: str) -> dict:
- """Load field definitions from JSON file."""
- if preset_type in _fields_cache:
- return _fields_cache[preset_type]
- # Map API type names to file names
- file_map = {
- "filament": "filament_fields.json",
- "print": "process_fields.json",
- "process": "process_fields.json",
- "printer": "printer_fields.json",
- }
- filename = file_map.get(preset_type)
- if not filename:
- raise HTTPException(status_code=400, detail=f"Unknown preset type: {preset_type}")
- file_path = FIELDS_DATA_DIR / filename
- if not file_path.exists():
- raise HTTPException(status_code=404, detail=f"Field definitions not found for: {preset_type}")
- with open(file_path) as f:
- data = json.load(f)
- _fields_cache[preset_type] = data
- return data
- @router.get("/builtin-filaments")
- async def get_builtin_filaments(
- _: User | None = RequirePermissionIfAuthEnabled(Permission.FILAMENTS_READ),
- ):
- """
- Get built-in filament names as a fallback source.
- Returns the static _BUILTIN_FILAMENT_NAMES table as a list of
- {filament_id, name} objects. Used by the frontend when cloud
- and local profiles are unavailable.
- """
- return [{"filament_id": fid, "name": name} for fid, name in _BUILTIN_FILAMENT_NAMES.items()]
- # Cache for filament_id → name mapping (resolved from cloud preset details)
- _filament_id_name_cache: dict[str, str] = {}
- _filament_id_name_cache_time: float = 0
- @router.get("/filament-id-map")
- async def get_filament_id_map(
- db: AsyncSession = Depends(get_db),
- current_user: User | None = RequirePermissionIfAuthEnabled(Permission.FILAMENTS_READ),
- ):
- """
- Get filament_id → name mapping for user cloud presets.
- K-profiles store a filament_id (e.g., "P4d64437") which is different from
- the cloud preset setting_id (e.g., "PFUS9ac902733670a9"). This endpoint
- fetches details for all custom presets and returns the mapping.
- Cached for 5 minutes.
- """
- import time
- global _filament_id_name_cache, _filament_id_name_cache_time
- if _filament_id_name_cache and time.time() - _filament_id_name_cache_time < FILAMENT_CACHE_TTL:
- return _filament_id_name_cache
- cloud = await build_authenticated_cloud(db, current_user)
- if cloud is None or not cloud.is_authenticated:
- if cloud is not None:
- await cloud.close()
- return _filament_id_name_cache or {}
- try:
- data = await cloud.get_slicer_settings()
- custom_presets = data.get("filament", {}).get("private", [])
- result: dict[str, str] = {}
- for preset in custom_presets:
- setting_id = preset.get("setting_id", "")
- if not setting_id:
- continue
- try:
- detail = await cloud.get_setting_detail(setting_id)
- fid = detail.get("filament_id", "")
- name = detail.get("name", "")
- if fid and name:
- # Strip printer/nozzle suffix: "Devil Design PLA Basic @Bambu Lab H2D 0.4 nozzle" → "Devil Design PLA Basic"
- clean_name = name.split(" @")[0].strip() if " @" in name else name
- result[fid] = clean_name
- except Exception:
- pass
- _filament_id_name_cache = result
- _filament_id_name_cache_time = time.time()
- return result
- except Exception:
- return _filament_id_name_cache or {}
- finally:
- await cloud.close()
- @router.get("/fields/{preset_type}")
- async def get_preset_fields(
- preset_type: Literal["filament", "print", "process", "printer"],
- _: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Get field definitions for a preset type.
- Returns a list of field definitions including:
- - key: The setting key name
- - label: Human-readable label
- - type: Field type (text, number, boolean, select)
- - category: Grouping category
- - description: Field description
- - options: For select fields, available options
- - unit: Unit of measurement (if applicable)
- - min/max/step: For number fields, validation constraints
- """
- data = _load_fields(preset_type)
- return data
- @router.get("/fields")
- async def get_all_preset_fields(
- _: User | None = RequirePermissionIfAuthEnabled(Permission.CLOUD_AUTH),
- ):
- """
- Get all field definitions for all preset types.
- Returns field definitions organized by type.
- """
- return {
- "filament": _load_fields("filament"),
- "process": _load_fields("process"),
- "printer": _load_fields("printer"),
- }
|