settings.py 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728
  1. import io
  2. import logging
  3. import zipfile
  4. from datetime import datetime
  5. from pathlib import Path
  6. from fastapi import APIRouter, Depends, File, UploadFile
  7. from fastapi.responses import JSONResponse, StreamingResponse
  8. from sqlalchemy import select
  9. from sqlalchemy.ext.asyncio import AsyncSession
  10. from backend.app.core.auth import RequirePermissionIfAuthEnabled
  11. from backend.app.core.config import settings as app_settings
  12. from backend.app.core.database import get_db
  13. from backend.app.core.permissions import Permission
  14. from backend.app.models.settings import Settings
  15. from backend.app.models.user import User
  16. from backend.app.schemas.settings import AppSettings, AppSettingsUpdate
  17. logger = logging.getLogger(__name__)
  18. router = APIRouter(prefix="/settings", tags=["settings"])
  19. # Default settings
  20. DEFAULT_SETTINGS = AppSettings()
  21. async def get_setting(db: AsyncSession, key: str) -> str | None:
  22. """Get a single setting value by key."""
  23. result = await db.execute(select(Settings).where(Settings.key == key))
  24. setting = result.scalar_one_or_none()
  25. return setting.value if setting else None
  26. async def get_external_login_url(db: AsyncSession) -> str:
  27. """Get the external URL for the login page.
  28. Uses external_url from settings if available, otherwise falls back to APP_URL env var.
  29. Args:
  30. db: Database session
  31. Returns:
  32. Full URL to the login page
  33. """
  34. import os
  35. external_url = await get_setting(db, "external_url")
  36. if external_url:
  37. external_url = external_url.rstrip("/")
  38. else:
  39. external_url = os.environ.get("APP_URL", "http://localhost:5173")
  40. return external_url + "/login"
  41. async def set_setting(db: AsyncSession, key: str, value: str) -> None:
  42. """Set a single setting value."""
  43. from sqlalchemy import func
  44. from sqlalchemy.dialects.sqlite import insert as sqlite_insert
  45. # Use upsert (INSERT ... ON CONFLICT UPDATE) for reliability
  46. stmt = sqlite_insert(Settings).values(key=key, value=value)
  47. stmt = stmt.on_conflict_do_update(index_elements=["key"], set_={"value": value, "updated_at": func.now()})
  48. await db.execute(stmt)
  49. @router.get("", response_model=AppSettings)
  50. @router.get("/", response_model=AppSettings)
  51. async def get_settings(
  52. db: AsyncSession = Depends(get_db),
  53. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_READ),
  54. ):
  55. """Get all application settings."""
  56. settings_dict = DEFAULT_SETTINGS.model_dump()
  57. # Load saved settings from database
  58. result = await db.execute(select(Settings))
  59. db_settings = result.scalars().all()
  60. for setting in db_settings:
  61. if setting.key in settings_dict:
  62. # Parse the value based on the expected type
  63. if setting.key in [
  64. "auto_archive",
  65. "save_thumbnails",
  66. "capture_finish_photo",
  67. "spoolman_enabled",
  68. "spoolman_disable_weight_sync",
  69. "spoolman_report_partial_usage",
  70. "check_updates",
  71. "check_printer_firmware",
  72. "virtual_printer_enabled",
  73. "ftp_retry_enabled",
  74. "mqtt_enabled",
  75. "mqtt_use_tls",
  76. "ha_enabled",
  77. "per_printer_mapping_expanded",
  78. "prometheus_enabled",
  79. ]:
  80. settings_dict[setting.key] = setting.value.lower() == "true"
  81. elif setting.key in [
  82. "default_filament_cost",
  83. "energy_cost_per_kwh",
  84. "ams_temp_good",
  85. "ams_temp_fair",
  86. "library_disk_warning_gb",
  87. ]:
  88. settings_dict[setting.key] = float(setting.value)
  89. elif setting.key in [
  90. "ams_humidity_good",
  91. "ams_humidity_fair",
  92. "ams_history_retention_days",
  93. "ftp_retry_count",
  94. "ftp_retry_delay",
  95. "ftp_timeout",
  96. "mqtt_port",
  97. ]:
  98. settings_dict[setting.key] = int(setting.value)
  99. elif setting.key == "default_printer_id":
  100. # Handle nullable integer
  101. settings_dict[setting.key] = int(setting.value) if setting.value and setting.value != "None" else None
  102. else:
  103. settings_dict[setting.key] = setting.value
  104. # Get Home Assistant settings (with environment variable overrides)
  105. ha_settings = await get_homeassistant_settings(db)
  106. settings_dict.update(ha_settings)
  107. return AppSettings(**settings_dict)
  108. @router.put("/", response_model=AppSettings)
  109. async def update_settings(
  110. settings_update: AppSettingsUpdate,
  111. db: AsyncSession = Depends(get_db),
  112. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_UPDATE),
  113. ):
  114. """Update application settings."""
  115. update_data = settings_update.model_dump(exclude_unset=True)
  116. # Check if any MQTT settings are being updated
  117. mqtt_keys = {
  118. "mqtt_enabled",
  119. "mqtt_broker",
  120. "mqtt_port",
  121. "mqtt_username",
  122. "mqtt_password",
  123. "mqtt_topic_prefix",
  124. "mqtt_use_tls",
  125. }
  126. mqtt_updated = bool(mqtt_keys & set(update_data.keys()))
  127. for key, value in update_data.items():
  128. # Convert value to string for storage
  129. if isinstance(value, bool):
  130. str_value = "true" if value else "false"
  131. elif value is None:
  132. str_value = "None"
  133. else:
  134. str_value = str(value)
  135. await set_setting(db, key, str_value)
  136. await db.commit()
  137. # Expire all objects to ensure fresh reads after commit
  138. db.expire_all()
  139. # Reconfigure MQTT relay if any MQTT settings changed
  140. if mqtt_updated:
  141. try:
  142. from backend.app.services.mqtt_relay import mqtt_relay
  143. mqtt_settings = {
  144. "mqtt_enabled": (await get_setting(db, "mqtt_enabled") or "false") == "true",
  145. "mqtt_broker": await get_setting(db, "mqtt_broker") or "",
  146. "mqtt_port": int(await get_setting(db, "mqtt_port") or "1883"),
  147. "mqtt_username": await get_setting(db, "mqtt_username") or "",
  148. "mqtt_password": await get_setting(db, "mqtt_password") or "",
  149. "mqtt_topic_prefix": await get_setting(db, "mqtt_topic_prefix") or "bambuddy",
  150. "mqtt_use_tls": (await get_setting(db, "mqtt_use_tls") or "false") == "true",
  151. }
  152. await mqtt_relay.configure(mqtt_settings)
  153. except Exception:
  154. pass # Don't fail the settings update if MQTT reconfiguration fails
  155. # Return updated settings
  156. return await get_settings(db)
  157. @router.patch("/", response_model=AppSettings)
  158. @router.patch("", response_model=AppSettings)
  159. async def patch_settings(
  160. settings_update: AppSettingsUpdate,
  161. db: AsyncSession = Depends(get_db),
  162. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_UPDATE),
  163. ):
  164. """Partially update application settings (same as PUT, for REST compatibility)."""
  165. return await update_settings(settings_update, db, _)
  166. @router.post("/reset", response_model=AppSettings)
  167. async def reset_settings(
  168. db: AsyncSession = Depends(get_db),
  169. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_UPDATE),
  170. ):
  171. """Reset all settings to defaults."""
  172. # Delete all settings
  173. result = await db.execute(select(Settings))
  174. for setting in result.scalars().all():
  175. await db.delete(setting)
  176. await db.commit()
  177. return DEFAULT_SETTINGS
  178. @router.get("/check-ffmpeg")
  179. async def check_ffmpeg():
  180. """Check if ffmpeg is installed and available."""
  181. from backend.app.services.camera import get_ffmpeg_path
  182. ffmpeg_path = get_ffmpeg_path()
  183. return {
  184. "installed": ffmpeg_path is not None,
  185. "path": ffmpeg_path,
  186. }
  187. @router.get("/spoolman")
  188. async def get_spoolman_settings(
  189. db: AsyncSession = Depends(get_db),
  190. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_READ),
  191. ):
  192. """Get Spoolman integration settings."""
  193. spoolman_enabled = await get_setting(db, "spoolman_enabled") or "false"
  194. spoolman_url = await get_setting(db, "spoolman_url") or ""
  195. spoolman_sync_mode = await get_setting(db, "spoolman_sync_mode") or "auto"
  196. spoolman_disable_weight_sync = await get_setting(db, "spoolman_disable_weight_sync") or "false"
  197. spoolman_report_partial_usage = await get_setting(db, "spoolman_report_partial_usage") or "true"
  198. return {
  199. "spoolman_enabled": spoolman_enabled,
  200. "spoolman_url": spoolman_url,
  201. "spoolman_sync_mode": spoolman_sync_mode,
  202. "spoolman_disable_weight_sync": spoolman_disable_weight_sync,
  203. "spoolman_report_partial_usage": spoolman_report_partial_usage,
  204. }
  205. @router.put("/spoolman")
  206. async def update_spoolman_settings(
  207. settings: dict,
  208. db: AsyncSession = Depends(get_db),
  209. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_UPDATE),
  210. ):
  211. """Update Spoolman integration settings."""
  212. if "spoolman_enabled" in settings:
  213. await set_setting(db, "spoolman_enabled", settings["spoolman_enabled"])
  214. if "spoolman_url" in settings:
  215. await set_setting(db, "spoolman_url", settings["spoolman_url"])
  216. if "spoolman_sync_mode" in settings:
  217. await set_setting(db, "spoolman_sync_mode", settings["spoolman_sync_mode"])
  218. if "spoolman_disable_weight_sync" in settings:
  219. await set_setting(db, "spoolman_disable_weight_sync", settings["spoolman_disable_weight_sync"])
  220. if "spoolman_report_partial_usage" in settings:
  221. await set_setting(db, "spoolman_report_partial_usage", settings["spoolman_report_partial_usage"])
  222. await db.commit()
  223. db.expire_all()
  224. # Return updated settings
  225. return await get_spoolman_settings(db)
  226. async def get_homeassistant_settings(db: AsyncSession) -> dict:
  227. """
  228. Get Home Assistant integration settings.
  229. Environment variables (HA_URL, HA_TOKEN) take precedence over database settings.
  230. """
  231. import os
  232. # Check environment variables first
  233. ha_url_env = os.environ.get("HA_URL")
  234. ha_token_env = os.environ.get("HA_TOKEN")
  235. # Fall back to database values
  236. ha_url = ha_url_env or await get_setting(db, "ha_url") or ""
  237. ha_token = ha_token_env or await get_setting(db, "ha_token") or ""
  238. ha_enabled_db = await get_setting(db, "ha_enabled") or "false"
  239. # Track which settings come from environment
  240. ha_url_from_env = bool(ha_url_env)
  241. ha_token_from_env = bool(ha_token_env)
  242. ha_env_managed = ha_url_from_env and ha_token_from_env
  243. # Auto-enable when both env vars are set, otherwise use database value
  244. if ha_url_env and ha_token_env:
  245. ha_enabled = True
  246. else:
  247. ha_enabled = ha_enabled_db.lower() == "true"
  248. return {
  249. "ha_enabled": ha_enabled,
  250. "ha_url": ha_url,
  251. "ha_token": ha_token,
  252. "ha_url_from_env": ha_url_from_env,
  253. "ha_token_from_env": ha_token_from_env,
  254. "ha_env_managed": ha_env_managed,
  255. }
  256. @router.get("/backup")
  257. async def create_backup(
  258. db: AsyncSession = Depends(get_db),
  259. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_BACKUP),
  260. ):
  261. """Create a complete backup (database + all files) as a ZIP.
  262. This is a simplified backup that includes the entire SQLite database
  263. and all data directories. It is complete by definition and cannot miss data.
  264. """
  265. import shutil
  266. import tempfile
  267. from sqlalchemy import text
  268. from backend.app.core.database import engine
  269. try:
  270. base_dir = app_settings.base_dir
  271. db_path = Path(app_settings.database_url.replace("sqlite+aiosqlite:///", ""))
  272. with tempfile.TemporaryDirectory() as temp_dir:
  273. temp_path = Path(temp_dir)
  274. # 1. Checkpoint WAL to ensure all data is in main db file
  275. async with engine.begin() as conn:
  276. await conn.execute(text("PRAGMA wal_checkpoint(TRUNCATE)"))
  277. # 2. Copy database file
  278. shutil.copy2(db_path, temp_path / "bambuddy.db")
  279. # 3. Copy data directories (if they exist)
  280. dirs_to_backup = [
  281. ("archive", base_dir / "archive"),
  282. ("virtual_printer", base_dir / "virtual_printer"),
  283. ("plate_calibration", app_settings.plate_calibration_dir),
  284. ("icons", base_dir / "icons"),
  285. ("projects", base_dir / "projects"),
  286. ]
  287. for name, src_dir in dirs_to_backup:
  288. if src_dir.exists() and any(src_dir.iterdir()):
  289. try:
  290. shutil.copytree(src_dir, temp_path / name)
  291. except shutil.Error as e:
  292. # Some files may have restricted permissions (e.g., SSL keys)
  293. # Log the error but continue with partial backup
  294. logger.warning("Some files in %s could not be copied: %s", name, e)
  295. except PermissionError as e:
  296. logger.warning("Permission denied copying %s: %s", name, e)
  297. # 4. Create ZIP
  298. zip_buffer = io.BytesIO()
  299. with zipfile.ZipFile(zip_buffer, "w", zipfile.ZIP_DEFLATED) as zf:
  300. for file_path in temp_path.rglob("*"):
  301. if file_path.is_file():
  302. arcname = file_path.relative_to(temp_path)
  303. zf.write(file_path, arcname)
  304. zip_buffer.seek(0)
  305. filename = f"bambuddy-backup-{datetime.now().strftime('%Y%m%d-%H%M%S')}.zip"
  306. return StreamingResponse(
  307. zip_buffer,
  308. media_type="application/zip",
  309. headers={"Content-Disposition": f"attachment; filename={filename}"},
  310. )
  311. except Exception as e:
  312. logger.error("Backup failed: %s", e, exc_info=True)
  313. return JSONResponse(
  314. status_code=500,
  315. content={"success": False, "message": "Backup failed. Check server logs for details."},
  316. )
  317. @router.post("/restore")
  318. async def restore_backup(
  319. file: UploadFile = File(...),
  320. db: AsyncSession = Depends(get_db),
  321. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_RESTORE),
  322. ):
  323. """Restore from a complete backup ZIP.
  324. This is a simplified restore that replaces the database and all data directories
  325. from the backup ZIP. Requires a restart after restore.
  326. """
  327. import shutil
  328. import tempfile
  329. from fastapi import HTTPException
  330. from backend.app.core.database import close_all_connections
  331. from backend.app.services.virtual_printer import virtual_printer_manager
  332. base_dir = app_settings.base_dir
  333. db_path = Path(app_settings.database_url.replace("sqlite+aiosqlite:///", ""))
  334. with tempfile.TemporaryDirectory() as temp_dir:
  335. temp_path = Path(temp_dir)
  336. # 1. Read and extract ZIP
  337. content = await file.read()
  338. # Check if it's a valid ZIP
  339. if not file.filename or not file.filename.endswith(".zip"):
  340. raise HTTPException(400, "Invalid backup file: must be a .zip file")
  341. try:
  342. with zipfile.ZipFile(io.BytesIO(content), "r") as zf:
  343. zf.extractall(temp_path)
  344. except zipfile.BadZipFile:
  345. raise HTTPException(400, "Invalid backup file: not a valid ZIP")
  346. # 2. Validate backup (must have database)
  347. backup_db = temp_path / "bambuddy.db"
  348. if not backup_db.exists():
  349. raise HTTPException(400, "Invalid backup: missing bambuddy.db")
  350. try:
  351. import asyncio
  352. # 3. Stop virtual printer if running (releases file locks)
  353. try:
  354. if virtual_printer_manager.is_enabled:
  355. logger.info("Stopping virtual printer for restore...")
  356. await virtual_printer_manager.configure(enabled=False)
  357. # Give it time to fully release file handles
  358. await asyncio.sleep(1)
  359. except Exception as e:
  360. logger.warning("Failed to stop virtual printer: %s", e)
  361. # 4. Close current database connections
  362. logger.info("Closing database connections...")
  363. await close_all_connections()
  364. # 5. Replace database
  365. logger.info("Restoring database from backup...")
  366. shutil.copy2(backup_db, db_path)
  367. # 6. Replace data directories
  368. # For Docker compatibility: clear contents then copy (don't delete mount points)
  369. dirs_to_restore = [
  370. ("archive", base_dir / "archive"),
  371. ("virtual_printer", base_dir / "virtual_printer"),
  372. ("plate_calibration", app_settings.plate_calibration_dir),
  373. ("icons", base_dir / "icons"),
  374. ("projects", base_dir / "projects"),
  375. ]
  376. skipped_dirs = []
  377. for name, dest_dir in dirs_to_restore:
  378. src_dir = temp_path / name
  379. if src_dir.exists():
  380. logger.info("Restoring %s directory...", name)
  381. try:
  382. # Clear destination contents (not the dir itself - may be Docker mount)
  383. if dest_dir.exists():
  384. for item in dest_dir.iterdir():
  385. try:
  386. if item.is_dir():
  387. shutil.rmtree(item)
  388. else:
  389. item.unlink()
  390. except OSError as e:
  391. logger.warning("Could not delete %s: %s", item, e)
  392. else:
  393. dest_dir.mkdir(parents=True, exist_ok=True)
  394. # Copy contents from backup
  395. for item in src_dir.iterdir():
  396. dest_item = dest_dir / item.name
  397. if item.is_dir():
  398. shutil.copytree(item, dest_item)
  399. else:
  400. shutil.copy2(item, dest_item)
  401. except OSError as e:
  402. logger.warning("Could not restore %s directory: %s", name, e)
  403. skipped_dirs.append(name)
  404. # 7. Note: Virtual printer and database will be reinitialized on restart
  405. # Do NOT try to restart services here - the database session is closed
  406. logger.info("Restore complete - restart required")
  407. message = "Backup restored successfully. Please restart Bambuddy for changes to take effect."
  408. if skipped_dirs:
  409. message += f" Note: Some directories could not be restored ({', '.join(skipped_dirs)})."
  410. return {
  411. "success": True,
  412. "message": message,
  413. }
  414. except Exception as e:
  415. logger.error("Restore failed: %s", e, exc_info=True)
  416. return JSONResponse(
  417. status_code=500,
  418. content={"success": False, "message": "Restore failed. Check server logs for details."},
  419. )
  420. @router.get("/network-interfaces")
  421. async def get_network_interfaces(
  422. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_READ),
  423. ):
  424. """Get available network interfaces for SSDP proxy configuration."""
  425. from backend.app.services.network_utils import get_network_interfaces
  426. interfaces = get_network_interfaces()
  427. return {"interfaces": interfaces}
  428. @router.get("/virtual-printer/models")
  429. async def get_virtual_printer_models(
  430. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_READ),
  431. ):
  432. """Get available virtual printer models."""
  433. from backend.app.services.virtual_printer import (
  434. DEFAULT_VIRTUAL_PRINTER_MODEL,
  435. VIRTUAL_PRINTER_MODELS,
  436. )
  437. return {
  438. "models": VIRTUAL_PRINTER_MODELS,
  439. "default": DEFAULT_VIRTUAL_PRINTER_MODEL,
  440. }
  441. @router.get("/virtual-printer")
  442. async def get_virtual_printer_settings(
  443. db: AsyncSession = Depends(get_db),
  444. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_READ),
  445. ):
  446. """Get virtual printer settings and status."""
  447. from backend.app.services.virtual_printer import (
  448. DEFAULT_VIRTUAL_PRINTER_MODEL,
  449. virtual_printer_manager,
  450. )
  451. enabled = await get_setting(db, "virtual_printer_enabled")
  452. access_code = await get_setting(db, "virtual_printer_access_code")
  453. mode = await get_setting(db, "virtual_printer_mode")
  454. model = await get_setting(db, "virtual_printer_model")
  455. target_printer_id = await get_setting(db, "virtual_printer_target_printer_id")
  456. remote_interface_ip = await get_setting(db, "virtual_printer_remote_interface_ip")
  457. return {
  458. "enabled": enabled == "true" if enabled else False,
  459. "access_code_set": bool(access_code),
  460. "mode": mode or "immediate",
  461. "model": model or DEFAULT_VIRTUAL_PRINTER_MODEL,
  462. "target_printer_id": int(target_printer_id) if target_printer_id else None,
  463. "remote_interface_ip": remote_interface_ip or "",
  464. "status": virtual_printer_manager.get_status(),
  465. }
  466. @router.put("/virtual-printer")
  467. async def update_virtual_printer_settings(
  468. enabled: bool = None,
  469. access_code: str = None,
  470. mode: str = None,
  471. model: str = None,
  472. target_printer_id: int = None,
  473. remote_interface_ip: str = None,
  474. db: AsyncSession = Depends(get_db),
  475. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_UPDATE),
  476. ):
  477. """Update virtual printer settings and restart services if needed.
  478. For proxy mode with SSDP proxy (dual-homed setup):
  479. - remote_interface_ip: IP of interface on slicer's network (LAN B)
  480. - Local interface is auto-detected based on target printer IP
  481. """
  482. from sqlalchemy import select
  483. from backend.app.models.printer import Printer
  484. from backend.app.services.virtual_printer import (
  485. DEFAULT_VIRTUAL_PRINTER_MODEL,
  486. VIRTUAL_PRINTER_MODELS,
  487. virtual_printer_manager,
  488. )
  489. # Get current values
  490. current_enabled = await get_setting(db, "virtual_printer_enabled") == "true"
  491. current_access_code = await get_setting(db, "virtual_printer_access_code") or ""
  492. current_mode = await get_setting(db, "virtual_printer_mode") or "immediate"
  493. current_model = await get_setting(db, "virtual_printer_model") or DEFAULT_VIRTUAL_PRINTER_MODEL
  494. current_target_id_str = await get_setting(db, "virtual_printer_target_printer_id")
  495. current_target_id = int(current_target_id_str) if current_target_id_str else None
  496. current_remote_iface = await get_setting(db, "virtual_printer_remote_interface_ip") or ""
  497. # Apply updates
  498. new_enabled = enabled if enabled is not None else current_enabled
  499. new_access_code = access_code if access_code is not None else current_access_code
  500. new_mode = mode if mode is not None else current_mode
  501. new_model = model if model is not None else current_model
  502. new_target_id = target_printer_id if target_printer_id is not None else current_target_id
  503. new_remote_iface = remote_interface_ip if remote_interface_ip is not None else current_remote_iface
  504. # Validate mode
  505. # "review" is the new name for "queue" (pending review before archiving)
  506. # "print_queue" archives and adds to print queue (unassigned)
  507. # "proxy" is transparent TCP proxy to a real printer
  508. if new_mode not in ("immediate", "queue", "review", "print_queue", "proxy"):
  509. return JSONResponse(
  510. status_code=400,
  511. content={"detail": "Mode must be 'immediate', 'review', 'print_queue', or 'proxy'"},
  512. )
  513. # Normalize legacy "queue" to "review" for storage
  514. if new_mode == "queue":
  515. new_mode = "review"
  516. # Validate model
  517. if model is not None and model not in VIRTUAL_PRINTER_MODELS:
  518. return JSONResponse(
  519. status_code=400,
  520. content={"detail": f"Invalid model. Must be one of: {', '.join(VIRTUAL_PRINTER_MODELS.keys())}"},
  521. )
  522. # Mode-specific validation and printer lookup
  523. target_printer_ip = ""
  524. target_printer_serial = ""
  525. if new_mode == "proxy":
  526. # Proxy mode requires target printer when enabling
  527. if new_enabled and not new_target_id:
  528. # If just switching to proxy mode (not explicitly enabling), auto-disable
  529. if enabled is None:
  530. new_enabled = False
  531. else:
  532. return JSONResponse(
  533. status_code=400,
  534. content={"detail": "Target printer is required for proxy mode"},
  535. )
  536. # Look up printer IP and serial if we have a target
  537. if new_target_id:
  538. result = await db.execute(select(Printer).where(Printer.id == new_target_id))
  539. printer = result.scalar_one_or_none()
  540. if not printer:
  541. return JSONResponse(
  542. status_code=400,
  543. content={"detail": f"Printer with ID {new_target_id} not found"},
  544. )
  545. target_printer_ip = printer.ip_address
  546. target_printer_serial = printer.serial_number
  547. # Access code not required for proxy mode
  548. else:
  549. # Non-proxy modes require access code when enabling
  550. if new_enabled and not new_access_code:
  551. # If just switching modes (not explicitly enabling), auto-disable
  552. if enabled is None:
  553. new_enabled = False
  554. else:
  555. return JSONResponse(
  556. status_code=400,
  557. content={"detail": "Access code is required when enabling virtual printer"},
  558. )
  559. # Validate access code length (Bambu Studio requires exactly 8 characters)
  560. if access_code is not None and access_code and len(access_code) != 8:
  561. return JSONResponse(
  562. status_code=400,
  563. content={"detail": "Access code must be exactly 8 characters"},
  564. )
  565. # Save settings
  566. await set_setting(db, "virtual_printer_enabled", "true" if new_enabled else "false")
  567. if access_code is not None:
  568. await set_setting(db, "virtual_printer_access_code", access_code)
  569. await set_setting(db, "virtual_printer_mode", new_mode)
  570. if model is not None:
  571. await set_setting(db, "virtual_printer_model", model)
  572. if target_printer_id is not None:
  573. await set_setting(db, "virtual_printer_target_printer_id", str(target_printer_id))
  574. if remote_interface_ip is not None:
  575. await set_setting(db, "virtual_printer_remote_interface_ip", remote_interface_ip)
  576. await db.commit()
  577. db.expire_all()
  578. # Reconfigure virtual printer
  579. try:
  580. await virtual_printer_manager.configure(
  581. enabled=new_enabled,
  582. access_code=new_access_code,
  583. mode=new_mode,
  584. model=new_model,
  585. target_printer_ip=target_printer_ip,
  586. target_printer_serial=target_printer_serial,
  587. remote_interface_ip=new_remote_iface,
  588. )
  589. except ValueError as e:
  590. logger.warning("Virtual printer configuration validation error: %s", e)
  591. return JSONResponse(
  592. status_code=400,
  593. content={"detail": "Invalid virtual printer configuration. Check the provided values."},
  594. )
  595. except Exception as e:
  596. logger.error("Failed to configure virtual printer: %s", e, exc_info=True)
  597. return JSONResponse(
  598. status_code=500,
  599. content={"detail": "Failed to configure virtual printer. Check server logs for details."},
  600. )
  601. return await get_virtual_printer_settings(db)
  602. # =============================================================================
  603. # MQTT Relay Settings
  604. # =============================================================================
  605. @router.get("/mqtt/status")
  606. async def get_mqtt_status(
  607. _: User | None = RequirePermissionIfAuthEnabled(Permission.SETTINGS_READ),
  608. ):
  609. """Get MQTT relay connection status."""
  610. from backend.app.services.mqtt_relay import mqtt_relay
  611. return mqtt_relay.get_status()