firmware_check.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589
  1. """
  2. Firmware Check Service
  3. Checks for firmware updates by fetching from Bambu Lab's official wiki and firmware
  4. download page. The wiki is used as the primary version source (always up-to-date),
  5. while the download page provides firmware file URLs for offline updates.
  6. """
  7. import logging
  8. import re
  9. import time
  10. from collections.abc import Callable
  11. from dataclasses import dataclass
  12. from pathlib import Path
  13. import httpx
  14. from backend.app.core.config import _data_dir
  15. logger = logging.getLogger(__name__)
  16. # Bambu Lab firmware download page (for download URLs)
  17. BAMBU_FIRMWARE_BASE = "https://bambulab.com"
  18. FIRMWARE_PAGE = "/en/support/firmware-download/all"
  19. # Bambu Lab wiki (primary source for latest version detection)
  20. BAMBU_WIKI_BASE = "https://wiki.bambulab.com"
  21. # Cache TTL in seconds (1 hour)
  22. CACHE_TTL = 3600
  23. # Map Bambuddy model names to Bambu Lab API keys
  24. MODEL_TO_API_KEY = {
  25. "X1": "x1",
  26. "X1C": "x1",
  27. "X1-Carbon": "x1",
  28. "X1 Carbon": "x1",
  29. "P1P": "p1",
  30. "P1S": "p1",
  31. "A1": "a1",
  32. "A1 Mini": "a1-mini",
  33. "A1-Mini": "a1-mini",
  34. "A1mini": "a1-mini",
  35. "H2D": "h2d",
  36. "H2C": "h2c",
  37. "H2S": "h2s",
  38. "P2S": "p2s",
  39. "X1E": "x1e",
  40. "X2D": "x2d",
  41. "H2D Pro": "h2d-pro",
  42. "H2D-Pro": "h2d-pro",
  43. "H2DPRO": "h2d-pro",
  44. # SSDP model codes (DevModel header) — in case raw codes are stored
  45. "O1D": "h2d",
  46. "O1E": "h2d-pro",
  47. "O2D": "h2d-pro",
  48. "O1C": "h2c",
  49. "O1C2": "h2c",
  50. "O1S": "h2s",
  51. "BL-P001": "x1",
  52. "BL-P002": "x1",
  53. "BL-P003": "x1e",
  54. "C11": "p1",
  55. "C12": "p1",
  56. "C13": "p2s",
  57. "N2S": "a1",
  58. "N1": "a1-mini",
  59. "N6": "x2d",
  60. "N7": "p2s",
  61. }
  62. # Reverse mapping: API key to model codes
  63. API_KEY_TO_DEV_MODEL = {
  64. "x1": "BL-P001",
  65. "p1": "C11",
  66. "a1": "N2S",
  67. "a1-mini": "N1",
  68. "h2d": "O1D",
  69. "h2c": "O1C",
  70. "h2s": "O1S",
  71. "p2s": "N7",
  72. "x1e": "C13",
  73. "x2d": "N6",
  74. "h2d-pro": "O1E",
  75. }
  76. # Wiki firmware release history pages (primary version source)
  77. API_KEY_TO_WIKI_PATH = {
  78. "x1": "/en/x1/manual/X1-X1C-firmware-release-history",
  79. "x1e": "/en/x1/manual/X1E-firmware-release-history",
  80. "p1": "/en/p1/manual/p1p-firmware-release-history",
  81. "a1": "/en/a1/manual/a1-firmware-release-history",
  82. "a1-mini": "/en/a1-mini/manual/a1-mini-firmware-release-history",
  83. "h2d": "/en/h2d/manual/h2d-firmware-release-history",
  84. "h2c": "/en/h2c/manual/h2c-firmware-release-history",
  85. "h2s": "/en/h2s/manual/h2s-firmware-release-history",
  86. "p2s": "/en/p2s/manual/p2s-firmware-release-history",
  87. "x2d": "/en/x2d/manual/x2d-firmware-release-history",
  88. "h2d-pro": "/en/h2d-pro/manual/firmware-release-history",
  89. }
  90. @dataclass
  91. class FirmwareVersion:
  92. """Firmware version information."""
  93. version: str
  94. download_url: str
  95. release_notes: str | None = None
  96. release_time: str | None = None
  97. class FirmwareCheckService:
  98. """Service for checking firmware updates from Bambu Lab."""
  99. def __init__(self):
  100. self._build_id: str | None = None
  101. self._build_id_time: float = 0
  102. self._version_cache: dict[str, FirmwareVersion] = {}
  103. self._versions_list_cache: dict[str, list[FirmwareVersion]] = {}
  104. self._cache_time: float = 0
  105. self._client = httpx.AsyncClient(
  106. timeout=30.0,
  107. headers={
  108. # Identify honestly as Bambuddy when scraping the public Bambu
  109. # Lab firmware wiki — verified 2026-05-12 that the wiki serves
  110. # this UA identically to a Chrome UA (same HTML response shape).
  111. # No browser impersonation needed for read-only public pages.
  112. "User-Agent": "Bambuddy/1.0 (+https://github.com/maziggy/bambuddy)"
  113. },
  114. )
  115. async def _get_build_id(self) -> str | None:
  116. """Fetch the Next.js build ID from Bambu Lab's firmware page."""
  117. # Use cached build ID if still valid (cache for 1 hour)
  118. if self._build_id and (time.time() - self._build_id_time) < CACHE_TTL:
  119. return self._build_id
  120. try:
  121. response = await self._client.get(f"{BAMBU_FIRMWARE_BASE}{FIRMWARE_PAGE}")
  122. if response.status_code == 200:
  123. # Extract buildId from the page
  124. match = re.search(r'"buildId":"([^"]+)"', response.text)
  125. if match:
  126. self._build_id = match.group(1)
  127. self._build_id_time = time.time()
  128. logger.info("Got Bambu Lab build ID: %s", self._build_id)
  129. return self._build_id
  130. logger.warning("Failed to get Bambu Lab page: %s", response.status_code)
  131. except Exception as e:
  132. logger.error("Error fetching Bambu Lab build ID: %s", e)
  133. return self._build_id # Return cached value if available
  134. async def _fetch_version_from_wiki(self, api_key: str) -> str | None:
  135. """Fetch the latest firmware version from Bambu Lab's wiki release history page."""
  136. versions = await self._fetch_all_versions_from_wiki(api_key)
  137. if versions:
  138. logger.debug("Wiki firmware for %s: %s", api_key, versions[0][0])
  139. return versions[0][0]
  140. return None
  141. async def _fetch_all_versions_from_wiki(self, api_key: str) -> list[tuple[str, str | None]]:
  142. """
  143. Fetch all firmware versions from the wiki release history page.
  144. Only extracts versions that appear in section-heading anchors
  145. (e.g. `id="h-01030000-20260303"` or `id="h-0102000020260409"`) —
  146. this excludes version-like numbers mentioned incidentally in
  147. release-note text. The dash separator between version and date is
  148. optional: H2D/X1/H2C/H2S still use it, but P2S and X2D publish
  149. anchors without the dash.
  150. Returns list of (version, release_date_YYYYMMDD | None) tuples, newest first.
  151. """
  152. wiki_path = API_KEY_TO_WIKI_PATH.get(api_key)
  153. if not wiki_path:
  154. return []
  155. try:
  156. url = f"{BAMBU_WIKI_BASE}{wiki_path}"
  157. response = await self._client.get(url, follow_redirects=True)
  158. if response.status_code != 200:
  159. return []
  160. # Primary: heading anchor ids like id="h-01030000-20260303" (dash)
  161. # or id="h-0102000020260409" (no dash, P2S/X2D-style).
  162. anchor_matches = re.findall(r'id="h-(\d{2})(\d{2})(\d{2})(\d{2})-?(\d{8})"', response.text)
  163. seen: set[str] = set()
  164. versions: list[tuple[str, str | None]] = []
  165. for a, b, c, d, date in anchor_matches:
  166. v = f"{a}.{b}.{c}.{d}"
  167. if v in seen:
  168. continue
  169. seen.add(v)
  170. versions.append((v, date))
  171. if versions:
  172. return versions
  173. # Fallback: heading text with "XX.XX.XX.XX (YYYYMMDD)" —
  174. # accept both ASCII "()" and full-width "()" (U+FF08/U+FF09)
  175. # which some pages (A1, A1-mini, P2S) use.
  176. text_matches = re.findall(
  177. r"(\d{2}\.\d{2}\.\d{2}\.\d{2})\s*[(\uff08](\d{8})[)\uff09]",
  178. response.text,
  179. )
  180. for v, date in text_matches:
  181. if v in seen:
  182. continue
  183. seen.add(v)
  184. versions.append((v, date))
  185. return versions
  186. except Exception as e:
  187. logger.debug("Error fetching wiki firmware list for %s: %s", api_key, e)
  188. return []
  189. async def _fetch_all_versions_from_download_page(self, api_key: str) -> list[FirmwareVersion]:
  190. """Fetch all firmware versions from Bambu Lab's download page (newest first)."""
  191. build_id = await self._get_build_id()
  192. if not build_id:
  193. return []
  194. try:
  195. url = f"{BAMBU_FIRMWARE_BASE}/_next/data/{build_id}/en/support/firmware-download/{api_key}.json"
  196. response = await self._client.get(url)
  197. if response.status_code == 200:
  198. data = response.json()
  199. page_props = data.get("pageProps", {})
  200. printer_map = page_props.get("printerMap", {})
  201. printer_data = printer_map.get(api_key, {})
  202. versions = printer_data.get("versions", [])
  203. return [
  204. FirmwareVersion(
  205. version=v.get("version", ""),
  206. download_url=v.get("url", ""),
  207. release_notes=v.get("release_notes_en"),
  208. release_time=v.get("release_time"),
  209. )
  210. for v in versions
  211. if v.get("version")
  212. ]
  213. except Exception as e:
  214. logger.debug("Error fetching download page firmware for %s: %s", api_key, e)
  215. return []
  216. async def _fetch_from_download_page(self, api_key: str) -> FirmwareVersion | None:
  217. """Fetch the latest firmware info from Bambu Lab's download page (has download URLs)."""
  218. versions = await self._fetch_all_versions_from_download_page(api_key)
  219. return versions[0] if versions else None
  220. async def _fetch_firmware_versions(self, api_key: str) -> FirmwareVersion | None:
  221. """Fetch firmware version info, using wiki as primary source and download page as fallback."""
  222. # Try wiki first (always has the latest version)
  223. wiki_version = await self._fetch_version_from_wiki(api_key)
  224. # Try download page (has download URLs, may lag behind wiki)
  225. download_info = await self._fetch_from_download_page(api_key)
  226. if wiki_version:
  227. # Wiki has the latest version — use it, attach download URL if available
  228. download_url = ""
  229. release_notes = None
  230. if download_info and download_info.version == wiki_version:
  231. download_url = download_info.download_url
  232. release_notes = download_info.release_notes
  233. return FirmwareVersion(
  234. version=wiki_version,
  235. download_url=download_url,
  236. release_notes=release_notes,
  237. )
  238. if download_info:
  239. return download_info
  240. logger.warning("Could not fetch firmware info for %s from wiki or download page", api_key)
  241. return None
  242. async def get_latest_version(self, model: str) -> FirmwareVersion | None:
  243. """
  244. Get the latest firmware version for a printer model.
  245. Args:
  246. model: Bambuddy printer model name (e.g., "X1C", "P1S", "H2D")
  247. Returns:
  248. FirmwareVersion if found, None otherwise
  249. """
  250. # Normalize model name
  251. model_upper = model.upper().replace(" ", "").replace("-", "")
  252. # Find the API key for this model
  253. api_key = None
  254. for model_name, key in MODEL_TO_API_KEY.items():
  255. if model_name.upper().replace(" ", "").replace("-", "") == model_upper:
  256. api_key = key
  257. break
  258. if not api_key:
  259. # Try direct lookup with original model
  260. api_key = MODEL_TO_API_KEY.get(model)
  261. if not api_key:
  262. logger.debug("Unknown printer model: %s", model)
  263. return None
  264. # Check cache
  265. cache_key = api_key
  266. if cache_key in self._version_cache and (time.time() - self._cache_time) < CACHE_TTL:
  267. return self._version_cache[cache_key]
  268. # Fetch from API
  269. version = await self._fetch_firmware_versions(api_key)
  270. if version:
  271. self._version_cache[cache_key] = version
  272. self._cache_time = time.time()
  273. return version
  274. def _resolve_api_key(self, model: str) -> str | None:
  275. """Resolve a model name to its Bambu API key."""
  276. model_upper = model.upper().replace(" ", "").replace("-", "")
  277. for name, key in MODEL_TO_API_KEY.items():
  278. if name.upper().replace(" ", "").replace("-", "") == model_upper:
  279. return key
  280. return MODEL_TO_API_KEY.get(model)
  281. @staticmethod
  282. def _version_tuple(v: str) -> tuple[int, ...]:
  283. parts = [int(x) for x in v.split(".")]
  284. while len(parts) < 4:
  285. parts.append(0)
  286. return tuple(parts)
  287. async def get_available_versions(self, model: str) -> list[FirmwareVersion]:
  288. """
  289. Get all announced firmware versions for a model, newest first.
  290. Merges the wiki release history (list of version strings) with the
  291. download page JSON (which provides download URLs + release notes).
  292. Versions present only on the wiki have an empty download_url and
  293. should be treated as "unavailable" for file-based installation.
  294. """
  295. api_key = self._resolve_api_key(model)
  296. if not api_key:
  297. return []
  298. if api_key in self._versions_list_cache and (time.time() - self._cache_time) < CACHE_TTL:
  299. return self._versions_list_cache[api_key]
  300. wiki_versions = await self._fetch_all_versions_from_wiki(api_key)
  301. download_versions = await self._fetch_all_versions_from_download_page(api_key)
  302. by_version: dict[str, FirmwareVersion] = {d.version: d for d in download_versions if d.version}
  303. merged: list[FirmwareVersion] = []
  304. seen: set[str] = set()
  305. for v, wiki_date in wiki_versions:
  306. if v in seen:
  307. continue
  308. seen.add(v)
  309. if v in by_version:
  310. merged.append(by_version[v])
  311. else:
  312. merged.append(FirmwareVersion(version=v, download_url="", release_time=wiki_date))
  313. for d in download_versions:
  314. if d.version and d.version not in seen:
  315. seen.add(d.version)
  316. merged.append(d)
  317. try:
  318. merged.sort(key=lambda fv: self._version_tuple(fv.version), reverse=True)
  319. except (ValueError, AttributeError):
  320. pass
  321. self._versions_list_cache[api_key] = merged
  322. self._cache_time = time.time()
  323. return merged
  324. async def get_version_info(self, model: str, version: str) -> FirmwareVersion | None:
  325. """Find a specific version's info (including download URL) for a model."""
  326. for v in await self.get_available_versions(model):
  327. if v.version == version:
  328. return v
  329. return None
  330. async def check_for_update(self, model: str, current_version: str) -> dict:
  331. """
  332. Check if a firmware update is available for a printer.
  333. Args:
  334. model: Printer model name
  335. current_version: Currently installed firmware version
  336. Returns:
  337. Dict with update info:
  338. - update_available: bool
  339. - current_version: str
  340. - latest_version: str or None
  341. - download_url: str or None
  342. - release_notes: str or None
  343. """
  344. result = {
  345. "update_available": False,
  346. "current_version": current_version,
  347. "latest_version": None,
  348. "download_url": None,
  349. "release_notes": None,
  350. "available_versions": [],
  351. }
  352. available = await self.get_available_versions(model)
  353. result["available_versions"] = [
  354. {
  355. "version": v.version,
  356. "download_url": v.download_url or None,
  357. "file_available": bool(v.download_url),
  358. "release_notes": v.release_notes,
  359. "release_time": v.release_time,
  360. }
  361. for v in available
  362. ]
  363. if not current_version:
  364. return result
  365. latest = available[0] if available else await self.get_latest_version(model)
  366. if not latest:
  367. return result
  368. result["latest_version"] = latest.version
  369. result["download_url"] = latest.download_url or None
  370. result["release_notes"] = latest.release_notes
  371. # Compare versions (format: XX.XX.XX.XX)
  372. try:
  373. current_parts = [int(x) for x in current_version.split(".")]
  374. latest_parts = [int(x) for x in latest.version.split(".")]
  375. # Pad to same length
  376. while len(current_parts) < 4:
  377. current_parts.append(0)
  378. while len(latest_parts) < 4:
  379. latest_parts.append(0)
  380. result["update_available"] = latest_parts > current_parts
  381. except (ValueError, AttributeError):
  382. logger.warning("Could not compare versions: %s vs %s", current_version, latest.version)
  383. return result
  384. async def get_all_latest_versions(self) -> dict[str, FirmwareVersion]:
  385. """
  386. Fetch latest firmware versions for all known printer models.
  387. Returns:
  388. Dict mapping API key to FirmwareVersion
  389. """
  390. results = {}
  391. for api_key in API_KEY_TO_DEV_MODEL:
  392. version = await self._fetch_firmware_versions(api_key)
  393. if version:
  394. results[api_key] = version
  395. return results
  396. def _get_firmware_cache_dir(self) -> Path:
  397. """Get the firmware cache directory, creating it if needed."""
  398. cache_dir = _data_dir / "firmware"
  399. cache_dir.mkdir(parents=True, exist_ok=True)
  400. return cache_dir
  401. async def get_firmware_file_info(self, model: str, version: str | None = None) -> dict | None:
  402. """
  403. Get information about a firmware file for a model.
  404. If `version` is provided, returns info for that specific version (must be
  405. available on the download page). Otherwise returns info for the latest version.
  406. """
  407. if version:
  408. target = await self.get_version_info(model, version)
  409. else:
  410. target = await self.get_latest_version(model)
  411. if not target or not target.download_url:
  412. return None
  413. url_parts = target.download_url.split("/")
  414. filename = url_parts[-1] if url_parts else f"firmware_{model}.bin"
  415. return {
  416. "download_url": target.download_url,
  417. "version": target.version,
  418. "filename": filename,
  419. "release_notes": target.release_notes,
  420. }
  421. async def download_firmware(
  422. self,
  423. model: str,
  424. progress_callback: Callable[[int, int, str], None] | None = None,
  425. version: str | None = None,
  426. ) -> Path | None:
  427. """
  428. Download firmware file for a printer model.
  429. Args:
  430. model: Printer model name (e.g., "X1C", "P1S", "H2D")
  431. progress_callback: Optional callback(bytes_downloaded, total_bytes, status_message)
  432. Returns:
  433. Path to downloaded firmware file, or None on failure
  434. """
  435. if version:
  436. latest = await self.get_version_info(model, version)
  437. else:
  438. latest = await self.get_latest_version(model)
  439. if not latest or not latest.download_url:
  440. logger.warning("No firmware download URL available for model %s version %s", model, version)
  441. return None
  442. # Extract original filename from URL (must preserve for SD card update)
  443. url_parts = latest.download_url.split("/")
  444. original_filename = url_parts[-1] if url_parts else f"firmware_{model}.bin"
  445. # Check if already cached (using original filename so SD card gets the right name)
  446. cached_path = self._get_firmware_cache_dir() / original_filename
  447. if cached_path.exists():
  448. logger.info("Using cached firmware: %s", cached_path)
  449. return cached_path
  450. # Download to temp file first
  451. temp_path = self._get_firmware_cache_dir() / f".downloading_{original_filename}"
  452. try:
  453. logger.info("Downloading firmware from %s", latest.download_url)
  454. if progress_callback:
  455. progress_callback(0, 0, "Starting download...")
  456. async with self._client.stream("GET", latest.download_url) as response:
  457. if response.status_code != 200:
  458. logger.error("Firmware download failed with status %s", response.status_code)
  459. return None
  460. total_size = int(response.headers.get("content-length", 0))
  461. downloaded = 0
  462. with open(temp_path, "wb") as f:
  463. async for chunk in response.aiter_bytes(chunk_size=65536):
  464. f.write(chunk)
  465. downloaded += len(chunk)
  466. if progress_callback:
  467. progress_callback(downloaded, total_size, "Downloading firmware...")
  468. # Move temp to final path, preserving original filename
  469. temp_path.rename(cached_path)
  470. logger.info("Firmware downloaded successfully: %s", cached_path)
  471. if progress_callback:
  472. progress_callback(downloaded, total_size, "Download complete")
  473. return cached_path
  474. except Exception as e:
  475. logger.error("Firmware download failed: %s", e)
  476. if temp_path.exists():
  477. try:
  478. temp_path.unlink()
  479. except OSError:
  480. pass # Best-effort cleanup of failed download temp file
  481. return None
  482. async def close(self):
  483. """Close the HTTP client."""
  484. await self._client.aclose()
  485. # Singleton instance
  486. _firmware_service: FirmwareCheckService | None = None
  487. def get_firmware_service() -> FirmwareCheckService:
  488. """Get the singleton firmware check service instance."""
  489. global _firmware_service
  490. if _firmware_service is None:
  491. _firmware_service = FirmwareCheckService()
  492. return _firmware_service