bambu_cloud.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552
  1. """
  2. Bambu Lab Cloud API Service
  3. Handles authentication and profile management with Bambu Lab's cloud services.
  4. """
  5. import logging
  6. from datetime import datetime, timedelta, timezone
  7. import httpx
  8. logger = logging.getLogger(__name__)
  9. BAMBU_API_BASE = "https://api.bambulab.com"
  10. BAMBU_API_BASE_CN = "https://api.bambulab.cn"
  11. class BambuCloudError(Exception):
  12. """Base exception for Bambu Cloud errors."""
  13. pass
  14. class BambuCloudAuthError(BambuCloudError):
  15. """Authentication related errors."""
  16. pass
  17. _shared_http_client: httpx.AsyncClient | None = None
  18. def set_shared_http_client(client: httpx.AsyncClient | None) -> None:
  19. """Register an app-scoped ``httpx.AsyncClient`` so per-request
  20. ``BambuCloudService`` instances can reuse its connection pool.
  21. Pass ``None`` during shutdown to unregister. The service only holds a
  22. reference (never closes a client it does not own), so region + token
  23. state still stays per-request — this only shares the transport pool.
  24. """
  25. global _shared_http_client
  26. _shared_http_client = client
  27. class BambuCloudService:
  28. """Service for interacting with Bambu Lab Cloud API."""
  29. def __init__(self, region: str = "global", client: httpx.AsyncClient | None = None):
  30. self.base_url = BAMBU_API_BASE if region == "global" else BAMBU_API_BASE_CN
  31. self.access_token: str | None = None
  32. self.refresh_token: str | None = None
  33. self.token_expiry: datetime | None = None
  34. # Prefer an explicitly-injected client (tests), else fall back to the
  35. # app-scoped shared client (production), and finally create our own so
  36. # scripts / tests that skip the lifespan still get a working service.
  37. if client is not None:
  38. self._client = client
  39. self._owns_client = False
  40. elif _shared_http_client is not None:
  41. self._client = _shared_http_client
  42. self._owns_client = False
  43. else:
  44. self._client = httpx.AsyncClient(timeout=30.0)
  45. self._owns_client = True
  46. @property
  47. def is_authenticated(self) -> bool:
  48. """Check if we have a valid token."""
  49. if not self.access_token:
  50. return False
  51. return not (self.token_expiry and datetime.now(timezone.utc) > self.token_expiry)
  52. def _get_headers(self) -> dict:
  53. """Get headers for authenticated requests."""
  54. headers = {
  55. "Content-Type": "application/json",
  56. "User-Agent": "Bambuddy/1.0",
  57. }
  58. if self.access_token:
  59. headers["Authorization"] = f"Bearer {self.access_token}"
  60. return headers
  61. async def login_request(self, email: str, password: str) -> dict:
  62. """
  63. Initiate login - this will trigger either email verification or TOTP prompt.
  64. Returns dict with login status, verification type, and tfaKey if needed.
  65. """
  66. try:
  67. response = await self._client.post(
  68. f"{self.base_url}/v1/user-service/user/login",
  69. headers={"Content-Type": "application/json"},
  70. json={
  71. "account": email,
  72. "password": password,
  73. },
  74. )
  75. data = response.json()
  76. logger.debug(
  77. f"Login response: status={response.status_code}, loginType={data.get('loginType')}, hasTfaKey={'tfaKey' in data}"
  78. )
  79. if response.status_code == 200:
  80. login_type = data.get("loginType")
  81. tfa_key = data.get("tfaKey")
  82. # TOTP authentication required
  83. if login_type == "tfa" or (tfa_key and login_type != "verifyCode"):
  84. return {
  85. "success": False,
  86. "needs_verification": True,
  87. "verification_type": "totp",
  88. "tfa_key": tfa_key,
  89. "message": "Enter the code from your authenticator app",
  90. }
  91. # Email verification required
  92. if login_type == "verifyCode":
  93. return {
  94. "success": False,
  95. "needs_verification": True,
  96. "verification_type": "email",
  97. "tfa_key": None,
  98. "message": "Verification code sent to email",
  99. }
  100. # Direct login success (rare, usually needs 2FA)
  101. if "accessToken" in data:
  102. self._set_tokens(data)
  103. return {"success": True, "needs_verification": False, "message": "Login successful"}
  104. # Handle specific error codes
  105. error_msg = data.get("message") or data.get("error") or "Login failed"
  106. return {"success": False, "needs_verification": False, "message": error_msg}
  107. except Exception as e:
  108. logger.error("Login request failed: %s", e)
  109. raise BambuCloudAuthError(f"Login request failed: {e}")
  110. async def verify_code(self, email: str, code: str) -> dict:
  111. """
  112. Complete login with email verification code.
  113. """
  114. try:
  115. response = await self._client.post(
  116. f"{self.base_url}/v1/user-service/user/login",
  117. headers={"Content-Type": "application/json"},
  118. json={
  119. "account": email,
  120. "code": code,
  121. },
  122. )
  123. data = response.json()
  124. logger.debug("Email verify response: status=%s, hasToken=%s", response.status_code, "accessToken" in data)
  125. if response.status_code == 200 and "accessToken" in data:
  126. self._set_tokens(data)
  127. return {"success": True, "message": "Login successful"}
  128. return {"success": False, "message": data.get("message", "Verification failed")}
  129. except Exception as e:
  130. logger.error("Email verification failed: %s", e)
  131. raise BambuCloudAuthError(f"Verification failed: {e}")
  132. async def verify_totp(self, tfa_key: str, code: str) -> dict:
  133. """
  134. Complete login with TOTP code from authenticator app.
  135. Args:
  136. tfa_key: The tfaKey returned from initial login request
  137. code: 6-digit TOTP code from authenticator app
  138. """
  139. try:
  140. # TFA endpoint is on bambulab.com, NOT api.bambulab.com
  141. # Requires browser-like headers to bypass Cloudflare
  142. tfa_url = "https://bambulab.com/api/sign-in/tfa"
  143. if "bambulab.cn" in self.base_url:
  144. tfa_url = "https://bambulab.cn/api/sign-in/tfa"
  145. browser_headers = {
  146. "Content-Type": "application/json",
  147. "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36",
  148. "Accept": "application/json, text/plain, */*",
  149. "Accept-Language": "en-US,en;q=0.9",
  150. "Origin": "https://bambulab.com",
  151. "Referer": "https://bambulab.com/",
  152. }
  153. response = await self._client.post(
  154. tfa_url,
  155. headers=browser_headers,
  156. json={
  157. "tfaKey": tfa_key,
  158. "tfaCode": code,
  159. },
  160. )
  161. logger.debug(
  162. f"TOTP verify response: status={response.status_code}, body={response.text[:200] if response.text else '(empty)'}"
  163. )
  164. # Handle empty response
  165. if not response.text or not response.text.strip():
  166. logger.warning("TOTP verification returned empty response (status %s)", response.status_code)
  167. return {"success": False, "message": "Bambu Cloud returned empty response. Please try again."}
  168. try:
  169. data = response.json()
  170. except Exception as json_err:
  171. logger.error("Failed to parse TOTP response: %s, body: %s", json_err, response.text[:500])
  172. return {"success": False, "message": "Invalid response from Bambu Cloud"}
  173. # Token might be in accessToken, token field, or cookies
  174. access_token = data.get("accessToken") or data.get("token")
  175. # Also check cookies for token
  176. if not access_token:
  177. for cookie in response.cookies:
  178. if "token" in cookie.lower():
  179. access_token = response.cookies.get(cookie)
  180. break
  181. if response.status_code == 200 and access_token:
  182. self.access_token = access_token
  183. self.refresh_token = data.get("refreshToken")
  184. from datetime import datetime, timedelta, timezone
  185. self.token_expiry = datetime.now(timezone.utc) + timedelta(days=30)
  186. return {"success": True, "message": "Login successful"}
  187. # Provide helpful error message
  188. error_msg = data.get("message", "")
  189. if "expired" in error_msg.lower():
  190. return {"success": False, "message": "TOTP session expired. Please try logging in again."}
  191. if not error_msg:
  192. error_msg = f"TOTP verification failed (status {response.status_code})"
  193. return {"success": False, "message": error_msg}
  194. except Exception as e:
  195. logger.error("TOTP verification failed: %s", e)
  196. # Return error instead of raising - don't trigger 401/500
  197. return {"success": False, "message": f"TOTP verification error: {e}"}
  198. def _set_tokens(self, data: dict):
  199. """Set tokens from login response."""
  200. self.access_token = data.get("accessToken")
  201. self.refresh_token = data.get("refreshToken")
  202. # Token typically valid for ~3 months, but we'll refresh more often
  203. self.token_expiry = datetime.now(timezone.utc) + timedelta(days=30)
  204. def set_token(self, access_token: str):
  205. """Set access token directly (for stored tokens)."""
  206. self.access_token = access_token
  207. self.token_expiry = datetime.now(timezone.utc) + timedelta(days=30)
  208. def logout(self):
  209. """Clear authentication state."""
  210. self.access_token = None
  211. self.refresh_token = None
  212. self.token_expiry = None
  213. async def get_user_profile(self) -> dict:
  214. """Get user profile information."""
  215. if not self.is_authenticated:
  216. raise BambuCloudAuthError("Not authenticated")
  217. try:
  218. response = await self._client.get(
  219. f"{self.base_url}/v1/design-user-service/my/preference", headers=self._get_headers()
  220. )
  221. if response.status_code == 200:
  222. return response.json()
  223. raise BambuCloudError(f"Failed to get profile: {response.status_code}")
  224. except httpx.RequestError as e:
  225. raise BambuCloudError(f"Request failed: {e}")
  226. async def get_slicer_settings(self, version: str = "02.04.00.70") -> dict:
  227. """
  228. Get all slicer settings (filament, printer, process presets).
  229. Args:
  230. version: Slicer version string
  231. """
  232. if not self.is_authenticated:
  233. raise BambuCloudAuthError("Not authenticated")
  234. try:
  235. response = await self._client.get(
  236. f"{self.base_url}/v1/iot-service/api/slicer/setting",
  237. headers=self._get_headers(),
  238. params={"version": version},
  239. )
  240. data = response.json()
  241. if response.status_code == 200:
  242. return data
  243. raise BambuCloudError(f"Failed to get settings: {response.status_code}")
  244. except httpx.RequestError as e:
  245. raise BambuCloudError(f"Request failed: {e}")
  246. async def get_setting_detail(self, setting_id: str) -> dict:
  247. """Get detailed information for a specific setting/preset."""
  248. if not self.is_authenticated:
  249. raise BambuCloudAuthError("Not authenticated")
  250. try:
  251. response = await self._client.get(
  252. f"{self.base_url}/v1/iot-service/api/slicer/setting/{setting_id}", headers=self._get_headers()
  253. )
  254. if response.status_code == 200:
  255. return response.json()
  256. raise BambuCloudError(f"Failed to get setting detail: {response.status_code}")
  257. except httpx.RequestError as e:
  258. raise BambuCloudError(f"Request failed: {e}")
  259. async def create_setting(
  260. self, preset_type: str, name: str, base_id: str, setting: dict, version: str = "2.0.0.0"
  261. ) -> dict:
  262. """
  263. Create a new slicer preset/setting.
  264. Args:
  265. preset_type: Type of preset - "filament", "print", or "printer"
  266. name: Display name for the preset
  267. base_id: Base preset ID to inherit from (e.g., "GFSA00")
  268. setting: Dict of setting key-value pairs (only modified values from base)
  269. version: Version string for the preset (default: "2.0.0.0")
  270. Returns:
  271. Created preset data including the new setting_id
  272. """
  273. if not self.is_authenticated:
  274. raise BambuCloudAuthError("Not authenticated")
  275. try:
  276. # Add timestamp if not present
  277. import time
  278. if "updated_time" not in setting:
  279. setting["updated_time"] = str(int(time.time()))
  280. payload = {
  281. "type": preset_type,
  282. "name": name,
  283. "version": version,
  284. "base_id": base_id,
  285. "setting": setting,
  286. }
  287. response = await self._client.post(
  288. f"{self.base_url}/v1/iot-service/api/slicer/setting", headers=self._get_headers(), json=payload
  289. )
  290. data = response.json()
  291. if response.status_code in (200, 201):
  292. return data
  293. error_msg = data.get("message") or data.get("error") or f"HTTP {response.status_code}"
  294. raise BambuCloudError(f"Failed to create setting: {error_msg}")
  295. except httpx.RequestError as e:
  296. raise BambuCloudError(f"Request failed: {e}")
  297. async def update_setting(self, setting_id: str, name: str | None = None, setting: dict | None = None) -> dict:
  298. """
  299. Update an existing slicer preset/setting.
  300. Note: Bambu Cloud API doesn't support true updates. Instead, we:
  301. 1. Fetch the current setting metadata (type, base_id, version)
  302. 2. Use the provided settings as the new complete settings (NOT merged)
  303. 3. Delete the old setting first (to avoid name conflicts)
  304. 4. Create a new setting via POST
  305. Args:
  306. setting_id: ID of the preset to update
  307. name: New display name (optional)
  308. setting: Dict of setting key-value pairs - this REPLACES the old settings entirely
  309. Returns:
  310. Updated preset data with new setting_id
  311. """
  312. if not self.is_authenticated:
  313. raise BambuCloudAuthError("Not authenticated")
  314. try:
  315. # Fetch current setting to get metadata (type, base_id, version)
  316. current = await self.get_setting_detail(setting_id)
  317. preset_type = current.get("type", "filament")
  318. # Use provided settings directly (complete replacement, not merge)
  319. # This allows the frontend to edit the full settings JSON
  320. if setting is not None:
  321. updated_setting = setting.copy()
  322. else:
  323. updated_setting = current.get("setting", {}).copy()
  324. # Extract name from settings_id field in the JSON, or use provided name, or fall back to current
  325. # The settings_id field contains the name in quotes, e.g., '"My Preset Name"'
  326. settings_id_key = {
  327. "filament": "filament_settings_id",
  328. "print": "print_settings_id",
  329. "printer": "printer_settings_id",
  330. }.get(preset_type, "filament_settings_id")
  331. settings_id_value = updated_setting.get(settings_id_key, "")
  332. if settings_id_value:
  333. # Remove surrounding quotes if present (e.g., '"foo"' -> 'foo')
  334. updated_name = settings_id_value.strip('"')
  335. elif name is not None:
  336. updated_name = name
  337. else:
  338. updated_name = current.get("name", "Untitled")
  339. # Update the timestamp
  340. import time
  341. updated_setting["updated_time"] = str(int(time.time()))
  342. # Ensure settings_id field matches the name
  343. updated_setting[settings_id_key] = f'"{updated_name}"'
  344. # Delete the old setting FIRST to avoid name conflicts
  345. await self.delete_setting(setting_id)
  346. # Create new setting via POST
  347. payload = {
  348. "type": preset_type,
  349. "name": updated_name,
  350. "version": current.get("version", "2.0.0.0"),
  351. "base_id": current.get("base_id", ""),
  352. "setting": updated_setting,
  353. }
  354. response = await self._client.post(
  355. f"{self.base_url}/v1/iot-service/api/slicer/setting", headers=self._get_headers(), json=payload
  356. )
  357. data = response.json()
  358. if response.status_code == 200:
  359. return data
  360. error_msg = data.get("message") or data.get("error") or f"HTTP {response.status_code}"
  361. raise BambuCloudError(f"Failed to update setting: {error_msg}")
  362. except httpx.RequestError as e:
  363. raise BambuCloudError(f"Request failed: {e}")
  364. async def delete_setting(self, setting_id: str) -> dict:
  365. """
  366. Delete a slicer preset/setting.
  367. Args:
  368. setting_id: ID of the preset to delete
  369. Returns:
  370. Deletion confirmation
  371. """
  372. if not self.is_authenticated:
  373. raise BambuCloudAuthError("Not authenticated")
  374. try:
  375. response = await self._client.delete(
  376. f"{self.base_url}/v1/iot-service/api/slicer/setting/{setting_id}", headers=self._get_headers()
  377. )
  378. if response.status_code in (200, 204):
  379. return {"success": True, "message": "Setting deleted"}
  380. data = response.json() if response.content else {}
  381. error_msg = data.get("message") or data.get("error") or f"HTTP {response.status_code}"
  382. raise BambuCloudError(f"Failed to delete setting: {error_msg}")
  383. except httpx.RequestError as e:
  384. raise BambuCloudError(f"Request failed: {e}")
  385. async def get_devices(self) -> dict:
  386. """Get list of bound devices."""
  387. if not self.is_authenticated:
  388. raise BambuCloudAuthError("Not authenticated")
  389. try:
  390. response = await self._client.get(
  391. f"{self.base_url}/v1/iot-service/api/user/bind", headers=self._get_headers()
  392. )
  393. if response.status_code == 200:
  394. return response.json()
  395. raise BambuCloudError(f"Failed to get devices: {response.status_code}")
  396. except httpx.RequestError as e:
  397. raise BambuCloudError(f"Request failed: {e}")
  398. async def get_firmware_version(self, device_id: str) -> dict:
  399. """
  400. Get firmware version info for a device.
  401. Returns dict with:
  402. - current_version: Installed firmware version
  403. - latest_version: Latest available firmware version
  404. - update_available: Boolean indicating if update is available
  405. - release_notes: Release notes for latest version
  406. """
  407. if not self.is_authenticated:
  408. raise BambuCloudAuthError("Not authenticated")
  409. try:
  410. response = await self._client.get(
  411. f"{self.base_url}/v1/iot-service/api/user/device/version",
  412. headers=self._get_headers(),
  413. params={"device_id": device_id},
  414. )
  415. if response.status_code == 200:
  416. data = response.json()
  417. # API wraps response in 'data' field
  418. return data.get("data", data)
  419. raise BambuCloudError(f"Failed to get firmware version: {response.status_code}")
  420. except httpx.RequestError as e:
  421. raise BambuCloudError(f"Request failed: {e}")
  422. async def close(self):
  423. """Close the HTTP client we own. No-op when sharing an app-scoped client."""
  424. if self._owns_client:
  425. await self._client.aclose()
  426. # Previously this module exposed a process-wide ``_cloud_service`` singleton
  427. # via ``get_cloud_service()`` / ``reset_cloud_service()``. That pattern leaked
  428. # region and token state across users (a China-region login would pin the
  429. # singleton to api.bambulab.cn until the next explicit reset), so the singleton
  430. # has been removed. Callers should construct a per-request
  431. # ``BambuCloudService(region=...)`` from the stored region and ``await
  432. # cloud.close()`` it when done. See ``routes.cloud.build_authenticated_cloud``
  433. # for the standard pattern.