cloud.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652
  1. """
  2. Bambu Lab Cloud API Routes
  3. Handles authentication and profile management with Bambu Cloud.
  4. """
  5. import json
  6. import logging
  7. from pathlib import Path
  8. from typing import Literal
  9. from fastapi import APIRouter, Body, Depends, HTTPException
  10. from sqlalchemy import select
  11. from sqlalchemy.ext.asyncio import AsyncSession
  12. from backend.app.core.database import get_db
  13. from backend.app.models.settings import Settings
  14. from backend.app.schemas.cloud import (
  15. CloudAuthStatus,
  16. CloudDevice,
  17. CloudLoginRequest,
  18. CloudLoginResponse,
  19. CloudTokenRequest,
  20. CloudVerifyRequest,
  21. FirmwareUpdateInfo,
  22. FirmwareUpdatesResponse,
  23. SlicerSetting,
  24. SlicerSettingCreate,
  25. SlicerSettingDeleteResponse,
  26. SlicerSettingsResponse,
  27. SlicerSettingUpdate,
  28. )
  29. from backend.app.services.bambu_cloud import (
  30. BambuCloudAuthError,
  31. BambuCloudError,
  32. get_cloud_service,
  33. )
  34. logger = logging.getLogger(__name__)
  35. router = APIRouter(prefix="/cloud", tags=["cloud"])
  36. # Keys for storing cloud credentials in settings
  37. CLOUD_TOKEN_KEY = "bambu_cloud_token"
  38. CLOUD_EMAIL_KEY = "bambu_cloud_email"
  39. async def get_stored_token(db: AsyncSession) -> tuple[str | None, str | None]:
  40. """Get stored cloud token and email from database."""
  41. result = await db.execute(select(Settings).where(Settings.key.in_([CLOUD_TOKEN_KEY, CLOUD_EMAIL_KEY])))
  42. settings = {s.key: s.value for s in result.scalars().all()}
  43. return settings.get(CLOUD_TOKEN_KEY), settings.get(CLOUD_EMAIL_KEY)
  44. async def store_token(db: AsyncSession, token: str, email: str) -> None:
  45. """Store cloud token and email in database."""
  46. for key, value in [(CLOUD_TOKEN_KEY, token), (CLOUD_EMAIL_KEY, email)]:
  47. result = await db.execute(select(Settings).where(Settings.key == key))
  48. setting = result.scalar_one_or_none()
  49. if setting:
  50. setting.value = value
  51. else:
  52. db.add(Settings(key=key, value=value))
  53. await db.commit()
  54. async def clear_token(db: AsyncSession) -> None:
  55. """Clear stored cloud token and email."""
  56. result = await db.execute(select(Settings).where(Settings.key.in_([CLOUD_TOKEN_KEY, CLOUD_EMAIL_KEY])))
  57. for setting in result.scalars().all():
  58. await db.delete(setting)
  59. await db.commit()
  60. @router.get("/status", response_model=CloudAuthStatus)
  61. async def get_auth_status(db: AsyncSession = Depends(get_db)):
  62. """Get current cloud authentication status."""
  63. token, email = await get_stored_token(db)
  64. cloud = get_cloud_service()
  65. if token:
  66. cloud.set_token(token)
  67. return CloudAuthStatus(
  68. is_authenticated=cloud.is_authenticated,
  69. email=email if cloud.is_authenticated else None,
  70. )
  71. @router.post("/login", response_model=CloudLoginResponse)
  72. async def login(request: CloudLoginRequest, db: AsyncSession = Depends(get_db)):
  73. """
  74. Initiate login to Bambu Cloud.
  75. This will typically trigger a verification code to be sent to the user's email.
  76. After receiving the code, call /cloud/verify to complete the login.
  77. """
  78. cloud = get_cloud_service()
  79. # Store email temporarily for verification step
  80. await store_token(db, "", request.email)
  81. try:
  82. result = await cloud.login_request(request.email, request.password)
  83. if result.get("success") and cloud.access_token:
  84. # Direct login succeeded (rare)
  85. await store_token(db, cloud.access_token, request.email)
  86. return CloudLoginResponse(
  87. success=result.get("success", False),
  88. needs_verification=result.get("needs_verification", False),
  89. message=result.get("message", "Unknown error"),
  90. )
  91. except BambuCloudAuthError as e:
  92. raise HTTPException(status_code=401, detail=str(e))
  93. except BambuCloudError as e:
  94. raise HTTPException(status_code=500, detail=str(e))
  95. @router.post("/verify", response_model=CloudLoginResponse)
  96. async def verify_code(request: CloudVerifyRequest, db: AsyncSession = Depends(get_db)):
  97. """
  98. Complete login with verification code.
  99. After calling /cloud/login, the user will receive an email with a 6-digit code.
  100. Submit that code here to complete authentication.
  101. """
  102. cloud = get_cloud_service()
  103. try:
  104. result = await cloud.verify_code(request.email, request.code)
  105. if result.get("success") and cloud.access_token:
  106. await store_token(db, cloud.access_token, request.email)
  107. return CloudLoginResponse(
  108. success=result.get("success", False),
  109. needs_verification=False,
  110. message=result.get("message", "Unknown error"),
  111. )
  112. except BambuCloudAuthError as e:
  113. raise HTTPException(status_code=401, detail=str(e))
  114. except BambuCloudError as e:
  115. raise HTTPException(status_code=500, detail=str(e))
  116. @router.post("/token", response_model=CloudAuthStatus)
  117. async def set_token(request: CloudTokenRequest, db: AsyncSession = Depends(get_db)):
  118. """
  119. Set access token directly.
  120. For users who already have a token (e.g., from Bambu Studio).
  121. """
  122. cloud = get_cloud_service()
  123. cloud.set_token(request.access_token)
  124. # Verify token works by trying to get profile
  125. try:
  126. await cloud.get_user_profile()
  127. await store_token(db, request.access_token, "token-auth")
  128. return CloudAuthStatus(is_authenticated=True, email="token-auth")
  129. except BambuCloudError:
  130. cloud.logout()
  131. raise HTTPException(status_code=401, detail="Invalid token")
  132. @router.post("/logout")
  133. async def logout(db: AsyncSession = Depends(get_db)):
  134. """Log out of Bambu Cloud."""
  135. cloud = get_cloud_service()
  136. cloud.logout()
  137. await clear_token(db)
  138. return {"success": True}
  139. @router.get("/settings", response_model=SlicerSettingsResponse)
  140. async def get_slicer_settings(
  141. version: str = "02.04.00.70",
  142. db: AsyncSession = Depends(get_db),
  143. ):
  144. """
  145. Get all slicer settings (filament, printer, process presets).
  146. Requires authentication.
  147. """
  148. token, _ = await get_stored_token(db)
  149. if not token:
  150. raise HTTPException(status_code=401, detail="Not authenticated")
  151. cloud = get_cloud_service()
  152. cloud.set_token(token)
  153. if not cloud.is_authenticated:
  154. raise HTTPException(status_code=401, detail="Not authenticated")
  155. try:
  156. data = await cloud.get_slicer_settings(version)
  157. result = SlicerSettingsResponse()
  158. # Map API keys to our types (API uses 'print' for process presets)
  159. type_mapping = {
  160. "filament": "filament",
  161. "printer": "printer",
  162. "print": "process", # API calls it 'print', we call it 'process'
  163. }
  164. for api_key, our_type in type_mapping.items():
  165. type_data = data.get(api_key, {})
  166. # Combine public and private presets, private (user's own) first
  167. all_settings = type_data.get("private", []) + type_data.get("public", [])
  168. parsed = []
  169. for s in all_settings:
  170. parsed.append(
  171. SlicerSetting(
  172. setting_id=s.get("setting_id", s.get("id", "")),
  173. name=s.get("name", "Unknown"),
  174. type=our_type,
  175. version=s.get("version"),
  176. user_id=s.get("user_id"),
  177. updated_time=s.get("updated_time"),
  178. )
  179. )
  180. setattr(result, our_type, parsed)
  181. return result
  182. except BambuCloudAuthError:
  183. await clear_token(db)
  184. raise HTTPException(status_code=401, detail="Authentication expired")
  185. except BambuCloudError as e:
  186. raise HTTPException(status_code=500, detail=str(e))
  187. @router.get("/settings/{setting_id}")
  188. async def get_setting_detail(setting_id: str, db: AsyncSession = Depends(get_db)):
  189. """
  190. Get detailed information for a specific setting/preset.
  191. Returns the full preset configuration.
  192. """
  193. token, _ = await get_stored_token(db)
  194. if not token:
  195. raise HTTPException(status_code=401, detail="Not authenticated")
  196. cloud = get_cloud_service()
  197. cloud.set_token(token)
  198. if not cloud.is_authenticated:
  199. raise HTTPException(status_code=401, detail="Not authenticated")
  200. try:
  201. data = await cloud.get_setting_detail(setting_id)
  202. return data
  203. except BambuCloudAuthError:
  204. await clear_token(db)
  205. raise HTTPException(status_code=401, detail="Authentication expired")
  206. except BambuCloudError as e:
  207. raise HTTPException(status_code=500, detail=str(e))
  208. # Cache for filament preset info (setting_id -> {name, k})
  209. _filament_cache: dict[str, dict] = {}
  210. _filament_cache_time: float = 0
  211. FILAMENT_CACHE_TTL = 300 # 5 minutes
  212. def _filament_id_to_setting_id(filament_id: str) -> str:
  213. """
  214. Convert filament_id to setting_id format for Bambu Cloud API.
  215. Printers report filament_id (e.g., GFA00, GFG02) but the API expects
  216. setting_id format which has an "S" inserted after "GF" (e.g., GFSA00, GFSG02).
  217. User presets (starting with "P") and already-correct IDs are returned unchanged.
  218. """
  219. if not filament_id:
  220. return filament_id
  221. # User presets start with "P" - leave unchanged
  222. if filament_id.startswith("P"):
  223. return filament_id
  224. # Official Bambu presets: GFx## -> GFSx##
  225. # Check if it matches the filament_id pattern (GF followed by letter and digits)
  226. if filament_id.startswith("GF") and len(filament_id) >= 4:
  227. # Check if it's already a setting_id (has S after GF)
  228. if filament_id[2] == "S":
  229. return filament_id
  230. # Insert "S" after "GF": GFA00 -> GFSA00
  231. return f"GFS{filament_id[2:]}"
  232. return filament_id
  233. @router.post("/filament-info")
  234. async def get_filament_info(setting_ids: list[str] = Body(...), db: AsyncSession = Depends(get_db)):
  235. """
  236. Get filament preset info (name and K value) for multiple setting IDs.
  237. Used to enrich AMS tray tooltips with cloud preset data.
  238. """
  239. import time
  240. logger.info(f"get_filament_info called with {len(setting_ids)} IDs: {setting_ids}")
  241. global _filament_cache, _filament_cache_time
  242. # Clear stale cache
  243. if time.time() - _filament_cache_time > FILAMENT_CACHE_TTL:
  244. _filament_cache = {}
  245. _filament_cache_time = time.time()
  246. token, _ = await get_stored_token(db)
  247. if not token:
  248. logger.info("get_filament_info: Not authenticated, returning empty")
  249. # Return empty results if not authenticated (graceful degradation)
  250. return {}
  251. cloud = get_cloud_service()
  252. cloud.set_token(token)
  253. if not cloud.is_authenticated:
  254. return {}
  255. result = {}
  256. for setting_id in setting_ids:
  257. if not setting_id:
  258. continue
  259. # Check cache first
  260. if setting_id in _filament_cache:
  261. result[setting_id] = _filament_cache[setting_id]
  262. continue
  263. try:
  264. # Transform filament_id to setting_id format (GFA00 -> GFSA00)
  265. api_setting_id = _filament_id_to_setting_id(setting_id)
  266. data = await cloud.get_setting_detail(api_setting_id)
  267. setting = data.get("setting", {})
  268. # Extract name (e.g., "Bambu PLA Basic Jade White")
  269. name = data.get("name", "")
  270. # Extract K value (pressure_advance)
  271. k_value = setting.get("pressure_advance")
  272. if k_value is not None:
  273. try:
  274. k_value = float(k_value)
  275. except (ValueError, TypeError):
  276. k_value = None
  277. info = {"name": name, "k": k_value}
  278. # Cache using original ID so frontend gets expected response
  279. _filament_cache[setting_id] = info
  280. result[setting_id] = info
  281. except Exception as e:
  282. logger.warning(
  283. f"Failed to get cloud preset {setting_id} (API ID: {_filament_id_to_setting_id(setting_id)}): {e}"
  284. )
  285. # Cache the failure to avoid repeated requests
  286. _filament_cache[setting_id] = {"name": "", "k": None}
  287. result[setting_id] = {"name": "", "k": None}
  288. return result
  289. @router.get("/devices", response_model=list[CloudDevice])
  290. async def get_devices(db: AsyncSession = Depends(get_db)):
  291. """
  292. Get list of bound printer devices.
  293. Returns printers registered to the user's Bambu account.
  294. """
  295. token, _ = await get_stored_token(db)
  296. if not token:
  297. raise HTTPException(status_code=401, detail="Not authenticated")
  298. cloud = get_cloud_service()
  299. cloud.set_token(token)
  300. if not cloud.is_authenticated:
  301. raise HTTPException(status_code=401, detail="Not authenticated")
  302. try:
  303. data = await cloud.get_devices()
  304. devices = data.get("devices", [])
  305. return [
  306. CloudDevice(
  307. dev_id=d.get("dev_id", ""),
  308. name=d.get("name", "Unknown"),
  309. dev_model_name=d.get("dev_model_name"),
  310. dev_product_name=d.get("dev_product_name"),
  311. online=d.get("online", False),
  312. )
  313. for d in devices
  314. ]
  315. except BambuCloudAuthError:
  316. await clear_token(db)
  317. raise HTTPException(status_code=401, detail="Authentication expired")
  318. except BambuCloudError as e:
  319. raise HTTPException(status_code=500, detail=str(e))
  320. @router.get("/firmware-updates", response_model=FirmwareUpdatesResponse)
  321. async def get_firmware_updates(db: AsyncSession = Depends(get_db)):
  322. """
  323. Check for firmware updates for all bound devices.
  324. Returns firmware version info for each device including:
  325. - Current installed version
  326. - Latest available version
  327. - Whether an update is available
  328. - Release notes for the latest version
  329. Requires cloud authentication.
  330. """
  331. token, _ = await get_stored_token(db)
  332. if not token:
  333. raise HTTPException(status_code=401, detail="Not authenticated")
  334. cloud = get_cloud_service()
  335. cloud.set_token(token)
  336. if not cloud.is_authenticated:
  337. raise HTTPException(status_code=401, detail="Not authenticated")
  338. try:
  339. # First get list of bound devices
  340. devices_data = await cloud.get_devices()
  341. devices = devices_data.get("devices", [])
  342. updates = []
  343. updates_available = 0
  344. # Check firmware for each device
  345. for device in devices:
  346. device_id = device.get("dev_id", "")
  347. device_name = device.get("name", "Unknown")
  348. try:
  349. firmware_info = await cloud.get_firmware_version(device_id)
  350. update_available = firmware_info.get("update_available", False)
  351. if update_available:
  352. updates_available += 1
  353. updates.append(
  354. FirmwareUpdateInfo(
  355. device_id=device_id,
  356. device_name=device_name,
  357. current_version=firmware_info.get("current_version"),
  358. latest_version=firmware_info.get("latest_version"),
  359. update_available=update_available,
  360. release_notes=firmware_info.get("release_notes"),
  361. )
  362. )
  363. except BambuCloudError as e:
  364. logger.warning(f"Failed to get firmware info for {device_name}: {e}")
  365. # Still include device but with unknown firmware status
  366. updates.append(
  367. FirmwareUpdateInfo(
  368. device_id=device_id,
  369. device_name=device_name,
  370. current_version=None,
  371. latest_version=None,
  372. update_available=False,
  373. release_notes=None,
  374. )
  375. )
  376. return FirmwareUpdatesResponse(updates=updates, updates_available=updates_available)
  377. except BambuCloudAuthError:
  378. await clear_token(db)
  379. raise HTTPException(status_code=401, detail="Authentication expired")
  380. except BambuCloudError as e:
  381. raise HTTPException(status_code=500, detail=str(e))
  382. @router.post("/settings")
  383. async def create_setting(request: SlicerSettingCreate, db: AsyncSession = Depends(get_db)):
  384. """
  385. Create a new slicer preset/setting.
  386. Creates a new preset on Bambu Cloud. The preset inherits from a base preset
  387. and only stores the delta (modified values).
  388. Type should be: 'filament', 'print', or 'printer'
  389. """
  390. token, _ = await get_stored_token(db)
  391. if not token:
  392. raise HTTPException(status_code=401, detail="Not authenticated")
  393. cloud = get_cloud_service()
  394. cloud.set_token(token)
  395. if not cloud.is_authenticated:
  396. raise HTTPException(status_code=401, detail="Not authenticated")
  397. try:
  398. data = await cloud.create_setting(
  399. preset_type=request.type,
  400. name=request.name,
  401. base_id=request.base_id,
  402. setting=request.setting,
  403. version=request.version,
  404. )
  405. return data
  406. except BambuCloudAuthError:
  407. await clear_token(db)
  408. raise HTTPException(status_code=401, detail="Authentication expired")
  409. except BambuCloudError as e:
  410. raise HTTPException(status_code=500, detail=str(e))
  411. @router.put("/settings/{setting_id}")
  412. async def update_setting(
  413. setting_id: str,
  414. request: SlicerSettingUpdate,
  415. db: AsyncSession = Depends(get_db),
  416. ):
  417. """
  418. Update an existing slicer preset/setting.
  419. Updates the preset's name and/or settings on Bambu Cloud.
  420. """
  421. token, _ = await get_stored_token(db)
  422. if not token:
  423. raise HTTPException(status_code=401, detail="Not authenticated")
  424. cloud = get_cloud_service()
  425. cloud.set_token(token)
  426. if not cloud.is_authenticated:
  427. raise HTTPException(status_code=401, detail="Not authenticated")
  428. try:
  429. data = await cloud.update_setting(
  430. setting_id=setting_id,
  431. name=request.name,
  432. setting=request.setting,
  433. )
  434. return data
  435. except BambuCloudAuthError:
  436. await clear_token(db)
  437. raise HTTPException(status_code=401, detail="Authentication expired")
  438. except BambuCloudError as e:
  439. raise HTTPException(status_code=500, detail=str(e))
  440. @router.delete("/settings/{setting_id}", response_model=SlicerSettingDeleteResponse)
  441. async def delete_setting(setting_id: str, db: AsyncSession = Depends(get_db)):
  442. """
  443. Delete a slicer preset/setting.
  444. Removes the preset from Bambu Cloud. This cannot be undone.
  445. """
  446. token, _ = await get_stored_token(db)
  447. if not token:
  448. raise HTTPException(status_code=401, detail="Not authenticated")
  449. cloud = get_cloud_service()
  450. cloud.set_token(token)
  451. if not cloud.is_authenticated:
  452. raise HTTPException(status_code=401, detail="Not authenticated")
  453. try:
  454. result = await cloud.delete_setting(setting_id)
  455. return SlicerSettingDeleteResponse(
  456. success=result.get("success", True),
  457. message=result.get("message", "Setting deleted"),
  458. )
  459. except BambuCloudAuthError:
  460. await clear_token(db)
  461. raise HTTPException(status_code=401, detail="Authentication expired")
  462. except BambuCloudError as e:
  463. raise HTTPException(status_code=500, detail=str(e))
  464. # Path to field definition files
  465. FIELDS_DATA_DIR = Path(__file__).parent.parent.parent / "data"
  466. # Cache for field definitions (loaded once)
  467. _fields_cache: dict[str, dict] = {}
  468. def _load_fields(preset_type: str) -> dict:
  469. """Load field definitions from JSON file."""
  470. if preset_type in _fields_cache:
  471. return _fields_cache[preset_type]
  472. # Map API type names to file names
  473. file_map = {
  474. "filament": "filament_fields.json",
  475. "print": "process_fields.json",
  476. "process": "process_fields.json",
  477. "printer": "printer_fields.json",
  478. }
  479. filename = file_map.get(preset_type)
  480. if not filename:
  481. raise HTTPException(status_code=400, detail=f"Unknown preset type: {preset_type}")
  482. file_path = FIELDS_DATA_DIR / filename
  483. if not file_path.exists():
  484. raise HTTPException(status_code=404, detail=f"Field definitions not found for: {preset_type}")
  485. with open(file_path) as f:
  486. data = json.load(f)
  487. _fields_cache[preset_type] = data
  488. return data
  489. @router.get("/fields/{preset_type}")
  490. async def get_preset_fields(preset_type: Literal["filament", "print", "process", "printer"]):
  491. """
  492. Get field definitions for a preset type.
  493. Returns a list of field definitions including:
  494. - key: The setting key name
  495. - label: Human-readable label
  496. - type: Field type (text, number, boolean, select)
  497. - category: Grouping category
  498. - description: Field description
  499. - options: For select fields, available options
  500. - unit: Unit of measurement (if applicable)
  501. - min/max/step: For number fields, validation constraints
  502. """
  503. data = _load_fields(preset_type)
  504. return data
  505. @router.get("/fields")
  506. async def get_all_preset_fields():
  507. """
  508. Get all field definitions for all preset types.
  509. Returns field definitions organized by type.
  510. """
  511. return {
  512. "filament": _load_fields("filament"),
  513. "process": _load_fields("process"),
  514. "printer": _load_fields("printer"),
  515. }