settings.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489
  1. import io
  2. import zipfile
  3. from datetime import datetime
  4. from pathlib import Path
  5. from fastapi import APIRouter, Depends, File, UploadFile
  6. from fastapi.responses import JSONResponse, StreamingResponse
  7. from sqlalchemy import select
  8. from sqlalchemy.ext.asyncio import AsyncSession
  9. from backend.app.core.config import settings as app_settings
  10. from backend.app.core.database import get_db
  11. from backend.app.models.settings import Settings
  12. from backend.app.schemas.settings import AppSettings, AppSettingsUpdate
  13. router = APIRouter(prefix="/settings", tags=["settings"])
  14. # Default settings
  15. DEFAULT_SETTINGS = AppSettings()
  16. async def get_setting(db: AsyncSession, key: str) -> str | None:
  17. """Get a single setting value by key."""
  18. result = await db.execute(select(Settings).where(Settings.key == key))
  19. setting = result.scalar_one_or_none()
  20. return setting.value if setting else None
  21. async def set_setting(db: AsyncSession, key: str, value: str) -> None:
  22. """Set a single setting value."""
  23. from sqlalchemy import func
  24. from sqlalchemy.dialects.sqlite import insert as sqlite_insert
  25. # Use upsert (INSERT ... ON CONFLICT UPDATE) for reliability
  26. stmt = sqlite_insert(Settings).values(key=key, value=value)
  27. stmt = stmt.on_conflict_do_update(index_elements=["key"], set_={"value": value, "updated_at": func.now()})
  28. await db.execute(stmt)
  29. @router.get("", response_model=AppSettings)
  30. @router.get("/", response_model=AppSettings)
  31. async def get_settings(db: AsyncSession = Depends(get_db)):
  32. """Get all application settings."""
  33. settings_dict = DEFAULT_SETTINGS.model_dump()
  34. # Load saved settings from database
  35. result = await db.execute(select(Settings))
  36. db_settings = result.scalars().all()
  37. for setting in db_settings:
  38. if setting.key in settings_dict:
  39. # Parse the value based on the expected type
  40. if setting.key in [
  41. "auto_archive",
  42. "save_thumbnails",
  43. "capture_finish_photo",
  44. "spoolman_enabled",
  45. "check_updates",
  46. "check_printer_firmware",
  47. "virtual_printer_enabled",
  48. "ftp_retry_enabled",
  49. "mqtt_enabled",
  50. "mqtt_use_tls",
  51. "ha_enabled",
  52. "per_printer_mapping_expanded",
  53. "prometheus_enabled",
  54. ]:
  55. settings_dict[setting.key] = setting.value.lower() == "true"
  56. elif setting.key in [
  57. "default_filament_cost",
  58. "energy_cost_per_kwh",
  59. "ams_temp_good",
  60. "ams_temp_fair",
  61. "library_disk_warning_gb",
  62. ]:
  63. settings_dict[setting.key] = float(setting.value)
  64. elif setting.key in [
  65. "ams_humidity_good",
  66. "ams_humidity_fair",
  67. "ams_history_retention_days",
  68. "ftp_retry_count",
  69. "ftp_retry_delay",
  70. "mqtt_port",
  71. ]:
  72. settings_dict[setting.key] = int(setting.value)
  73. elif setting.key == "default_printer_id":
  74. # Handle nullable integer
  75. settings_dict[setting.key] = int(setting.value) if setting.value and setting.value != "None" else None
  76. else:
  77. settings_dict[setting.key] = setting.value
  78. return AppSettings(**settings_dict)
  79. @router.put("/", response_model=AppSettings)
  80. async def update_settings(
  81. settings_update: AppSettingsUpdate,
  82. db: AsyncSession = Depends(get_db),
  83. ):
  84. """Update application settings."""
  85. update_data = settings_update.model_dump(exclude_unset=True)
  86. # Check if any MQTT settings are being updated
  87. mqtt_keys = {
  88. "mqtt_enabled",
  89. "mqtt_broker",
  90. "mqtt_port",
  91. "mqtt_username",
  92. "mqtt_password",
  93. "mqtt_topic_prefix",
  94. "mqtt_use_tls",
  95. }
  96. mqtt_updated = bool(mqtt_keys & set(update_data.keys()))
  97. for key, value in update_data.items():
  98. # Convert value to string for storage
  99. if isinstance(value, bool):
  100. str_value = "true" if value else "false"
  101. elif value is None:
  102. str_value = "None"
  103. else:
  104. str_value = str(value)
  105. await set_setting(db, key, str_value)
  106. await db.commit()
  107. # Expire all objects to ensure fresh reads after commit
  108. db.expire_all()
  109. # Reconfigure MQTT relay if any MQTT settings changed
  110. if mqtt_updated:
  111. try:
  112. from backend.app.services.mqtt_relay import mqtt_relay
  113. mqtt_settings = {
  114. "mqtt_enabled": (await get_setting(db, "mqtt_enabled") or "false") == "true",
  115. "mqtt_broker": await get_setting(db, "mqtt_broker") or "",
  116. "mqtt_port": int(await get_setting(db, "mqtt_port") or "1883"),
  117. "mqtt_username": await get_setting(db, "mqtt_username") or "",
  118. "mqtt_password": await get_setting(db, "mqtt_password") or "",
  119. "mqtt_topic_prefix": await get_setting(db, "mqtt_topic_prefix") or "bambuddy",
  120. "mqtt_use_tls": (await get_setting(db, "mqtt_use_tls") or "false") == "true",
  121. }
  122. await mqtt_relay.configure(mqtt_settings)
  123. except Exception:
  124. pass # Don't fail the settings update if MQTT reconfiguration fails
  125. # Return updated settings
  126. return await get_settings(db)
  127. @router.patch("/", response_model=AppSettings)
  128. @router.patch("", response_model=AppSettings)
  129. async def patch_settings(
  130. settings_update: AppSettingsUpdate,
  131. db: AsyncSession = Depends(get_db),
  132. ):
  133. """Partially update application settings (same as PUT, for REST compatibility)."""
  134. return await update_settings(settings_update, db)
  135. @router.post("/reset", response_model=AppSettings)
  136. async def reset_settings(db: AsyncSession = Depends(get_db)):
  137. """Reset all settings to defaults."""
  138. # Delete all settings
  139. result = await db.execute(select(Settings))
  140. for setting in result.scalars().all():
  141. await db.delete(setting)
  142. await db.commit()
  143. return DEFAULT_SETTINGS
  144. @router.get("/check-ffmpeg")
  145. async def check_ffmpeg():
  146. """Check if ffmpeg is installed and available."""
  147. from backend.app.services.camera import get_ffmpeg_path
  148. ffmpeg_path = get_ffmpeg_path()
  149. return {
  150. "installed": ffmpeg_path is not None,
  151. "path": ffmpeg_path,
  152. }
  153. @router.get("/spoolman")
  154. async def get_spoolman_settings(db: AsyncSession = Depends(get_db)):
  155. """Get Spoolman integration settings."""
  156. spoolman_enabled = await get_setting(db, "spoolman_enabled") or "false"
  157. spoolman_url = await get_setting(db, "spoolman_url") or ""
  158. spoolman_sync_mode = await get_setting(db, "spoolman_sync_mode") or "auto"
  159. return {
  160. "spoolman_enabled": spoolman_enabled,
  161. "spoolman_url": spoolman_url,
  162. "spoolman_sync_mode": spoolman_sync_mode,
  163. }
  164. @router.put("/spoolman")
  165. async def update_spoolman_settings(
  166. settings: dict,
  167. db: AsyncSession = Depends(get_db),
  168. ):
  169. """Update Spoolman integration settings."""
  170. if "spoolman_enabled" in settings:
  171. await set_setting(db, "spoolman_enabled", settings["spoolman_enabled"])
  172. if "spoolman_url" in settings:
  173. await set_setting(db, "spoolman_url", settings["spoolman_url"])
  174. if "spoolman_sync_mode" in settings:
  175. await set_setting(db, "spoolman_sync_mode", settings["spoolman_sync_mode"])
  176. await db.commit()
  177. db.expire_all()
  178. # Return updated settings
  179. return await get_spoolman_settings(db)
  180. @router.get("/backup")
  181. async def create_backup(db: AsyncSession = Depends(get_db)):
  182. """Create a complete backup (database + all files) as a ZIP.
  183. This is a simplified backup that includes the entire SQLite database
  184. and all data directories. It is complete by definition and cannot miss data.
  185. """
  186. import shutil
  187. import tempfile
  188. from sqlalchemy import text
  189. from backend.app.core.database import engine
  190. base_dir = app_settings.base_dir
  191. db_path = Path(app_settings.database_url.replace("sqlite+aiosqlite:///", ""))
  192. with tempfile.TemporaryDirectory() as temp_dir:
  193. temp_path = Path(temp_dir)
  194. # 1. Checkpoint WAL to ensure all data is in main db file
  195. async with engine.begin() as conn:
  196. await conn.execute(text("PRAGMA wal_checkpoint(TRUNCATE)"))
  197. # 2. Copy database file
  198. shutil.copy2(db_path, temp_path / "bambuddy.db")
  199. # 3. Copy data directories (if they exist)
  200. dirs_to_backup = [
  201. ("archive", base_dir / "archive"),
  202. ("virtual_printer", base_dir / "virtual_printer"),
  203. ("plate_calibration", app_settings.plate_calibration_dir),
  204. ("icons", base_dir / "icons"),
  205. ("projects", base_dir / "projects"),
  206. ]
  207. for name, src_dir in dirs_to_backup:
  208. if src_dir.exists() and any(src_dir.iterdir()):
  209. shutil.copytree(src_dir, temp_path / name)
  210. # 4. Create ZIP
  211. zip_buffer = io.BytesIO()
  212. with zipfile.ZipFile(zip_buffer, "w", zipfile.ZIP_DEFLATED) as zf:
  213. for file_path in temp_path.rglob("*"):
  214. if file_path.is_file():
  215. arcname = file_path.relative_to(temp_path)
  216. zf.write(file_path, arcname)
  217. zip_buffer.seek(0)
  218. filename = f"bambuddy-backup-{datetime.now().strftime('%Y%m%d-%H%M%S')}.zip"
  219. return StreamingResponse(
  220. zip_buffer,
  221. media_type="application/zip",
  222. headers={"Content-Disposition": f"attachment; filename={filename}"},
  223. )
  224. @router.post("/restore")
  225. async def restore_backup(
  226. file: UploadFile = File(...),
  227. db: AsyncSession = Depends(get_db),
  228. ):
  229. """Restore from a complete backup ZIP.
  230. This is a simplified restore that replaces the database and all data directories
  231. from the backup ZIP. Requires a restart after restore.
  232. """
  233. import shutil
  234. import tempfile
  235. from fastapi import HTTPException
  236. from backend.app.core.database import close_all_connections
  237. base_dir = app_settings.base_dir
  238. db_path = Path(app_settings.database_url.replace("sqlite+aiosqlite:///", ""))
  239. with tempfile.TemporaryDirectory() as temp_dir:
  240. temp_path = Path(temp_dir)
  241. # 1. Read and extract ZIP
  242. content = await file.read()
  243. # Check if it's a valid ZIP
  244. if not file.filename or not file.filename.endswith(".zip"):
  245. raise HTTPException(400, "Invalid backup file: must be a .zip file")
  246. try:
  247. with zipfile.ZipFile(io.BytesIO(content), "r") as zf:
  248. zf.extractall(temp_path)
  249. except zipfile.BadZipFile:
  250. raise HTTPException(400, "Invalid backup file: not a valid ZIP")
  251. # 2. Validate backup (must have database)
  252. backup_db = temp_path / "bambuddy.db"
  253. if not backup_db.exists():
  254. raise HTTPException(400, "Invalid backup: missing bambuddy.db")
  255. # 3. Close current database connections
  256. await close_all_connections()
  257. # 4. Replace database
  258. shutil.copy2(backup_db, db_path)
  259. # 5. Replace data directories
  260. dirs_to_restore = [
  261. ("archive", base_dir / "archive"),
  262. ("virtual_printer", base_dir / "virtual_printer"),
  263. ("plate_calibration", app_settings.plate_calibration_dir),
  264. ("icons", base_dir / "icons"),
  265. ("projects", base_dir / "projects"),
  266. ]
  267. for name, dest_dir in dirs_to_restore:
  268. src_dir = temp_path / name
  269. if src_dir.exists():
  270. if dest_dir.exists():
  271. shutil.rmtree(dest_dir)
  272. shutil.copytree(src_dir, dest_dir)
  273. # 6. Note: Database connection will be reinitialized on restart
  274. # The application should be restarted after restore
  275. return {
  276. "success": True,
  277. "message": "Backup restored successfully. Please restart Bambuddy for changes to take effect.",
  278. }
  279. @router.get("/virtual-printer/models")
  280. async def get_virtual_printer_models():
  281. """Get available virtual printer models."""
  282. from backend.app.services.virtual_printer import (
  283. DEFAULT_VIRTUAL_PRINTER_MODEL,
  284. VIRTUAL_PRINTER_MODELS,
  285. )
  286. return {
  287. "models": VIRTUAL_PRINTER_MODELS,
  288. "default": DEFAULT_VIRTUAL_PRINTER_MODEL,
  289. }
  290. @router.get("/virtual-printer")
  291. async def get_virtual_printer_settings(db: AsyncSession = Depends(get_db)):
  292. """Get virtual printer settings and status."""
  293. from backend.app.services.virtual_printer import (
  294. DEFAULT_VIRTUAL_PRINTER_MODEL,
  295. virtual_printer_manager,
  296. )
  297. enabled = await get_setting(db, "virtual_printer_enabled")
  298. access_code = await get_setting(db, "virtual_printer_access_code")
  299. mode = await get_setting(db, "virtual_printer_mode")
  300. model = await get_setting(db, "virtual_printer_model")
  301. return {
  302. "enabled": enabled == "true" if enabled else False,
  303. "access_code_set": bool(access_code),
  304. "mode": mode or "immediate",
  305. "model": model or DEFAULT_VIRTUAL_PRINTER_MODEL,
  306. "status": virtual_printer_manager.get_status(),
  307. }
  308. @router.put("/virtual-printer")
  309. async def update_virtual_printer_settings(
  310. enabled: bool = None,
  311. access_code: str = None,
  312. mode: str = None,
  313. model: str = None,
  314. db: AsyncSession = Depends(get_db),
  315. ):
  316. """Update virtual printer settings and restart services if needed."""
  317. from backend.app.services.virtual_printer import (
  318. DEFAULT_VIRTUAL_PRINTER_MODEL,
  319. VIRTUAL_PRINTER_MODELS,
  320. virtual_printer_manager,
  321. )
  322. # Get current values
  323. current_enabled = await get_setting(db, "virtual_printer_enabled") == "true"
  324. current_access_code = await get_setting(db, "virtual_printer_access_code") or ""
  325. current_mode = await get_setting(db, "virtual_printer_mode") or "immediate"
  326. current_model = await get_setting(db, "virtual_printer_model") or DEFAULT_VIRTUAL_PRINTER_MODEL
  327. # Apply updates
  328. new_enabled = enabled if enabled is not None else current_enabled
  329. new_access_code = access_code if access_code is not None else current_access_code
  330. new_mode = mode if mode is not None else current_mode
  331. new_model = model if model is not None else current_model
  332. # Validate mode
  333. # "review" is the new name for "queue" (pending review before archiving)
  334. # "print_queue" archives and adds to print queue (unassigned)
  335. if new_mode not in ("immediate", "queue", "review", "print_queue"):
  336. return JSONResponse(
  337. status_code=400,
  338. content={"detail": "Mode must be 'immediate', 'review', or 'print_queue'"},
  339. )
  340. # Normalize legacy "queue" to "review" for storage
  341. if new_mode == "queue":
  342. new_mode = "review"
  343. # Validate model
  344. if model is not None and model not in VIRTUAL_PRINTER_MODELS:
  345. return JSONResponse(
  346. status_code=400,
  347. content={"detail": f"Invalid model. Must be one of: {', '.join(VIRTUAL_PRINTER_MODELS.keys())}"},
  348. )
  349. # Validate access code when enabling
  350. if new_enabled and not new_access_code:
  351. return JSONResponse(
  352. status_code=400,
  353. content={"detail": "Access code is required when enabling virtual printer"},
  354. )
  355. # Validate access code length (Bambu Studio requires exactly 8 characters)
  356. if access_code is not None and len(access_code) != 8:
  357. return JSONResponse(
  358. status_code=400,
  359. content={"detail": "Access code must be exactly 8 characters"},
  360. )
  361. # Save settings
  362. await set_setting(db, "virtual_printer_enabled", "true" if new_enabled else "false")
  363. if access_code is not None:
  364. await set_setting(db, "virtual_printer_access_code", access_code)
  365. await set_setting(db, "virtual_printer_mode", new_mode)
  366. if model is not None:
  367. await set_setting(db, "virtual_printer_model", model)
  368. await db.commit()
  369. db.expire_all()
  370. # Reconfigure virtual printer
  371. try:
  372. await virtual_printer_manager.configure(
  373. enabled=new_enabled,
  374. access_code=new_access_code,
  375. mode=new_mode,
  376. model=new_model,
  377. )
  378. except ValueError as e:
  379. return JSONResponse(
  380. status_code=400,
  381. content={"detail": str(e)},
  382. )
  383. except Exception as e:
  384. return JSONResponse(
  385. status_code=500,
  386. content={"detail": f"Failed to configure virtual printer: {e}"},
  387. )
  388. return await get_virtual_printer_settings(db)
  389. # =============================================================================
  390. # MQTT Relay Settings
  391. # =============================================================================
  392. @router.get("/mqtt/status")
  393. async def get_mqtt_status():
  394. """Get MQTT relay connection status."""
  395. from backend.app.services.mqtt_relay import mqtt_relay
  396. return mqtt_relay.get_status()