notifications.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472
  1. """API routes for notification providers."""
  2. import json
  3. import logging
  4. from datetime import datetime, timedelta, timezone
  5. from fastapi import APIRouter, Depends, HTTPException, Query
  6. from sqlalchemy import delete, desc, func, select
  7. from sqlalchemy.ext.asyncio import AsyncSession
  8. from backend.app.core.auth import RequirePermissionIfAuthEnabled
  9. from backend.app.core.database import get_db
  10. from backend.app.core.permissions import Permission
  11. from backend.app.models.notification import NotificationLog, NotificationProvider
  12. from backend.app.models.user import User
  13. from backend.app.schemas.notification import (
  14. NotificationLogResponse,
  15. NotificationLogStats,
  16. NotificationProviderCreate,
  17. NotificationProviderResponse,
  18. NotificationProviderUpdate,
  19. NotificationTestRequest,
  20. NotificationTestResponse,
  21. )
  22. from backend.app.services.notification_service import notification_service
  23. logger = logging.getLogger(__name__)
  24. router = APIRouter(prefix="/notifications", tags=["notifications"])
  25. def _provider_to_dict(provider: NotificationProvider) -> dict:
  26. """Convert a NotificationProvider model to a response dictionary."""
  27. return {
  28. "id": provider.id,
  29. "name": provider.name,
  30. "provider_type": provider.provider_type,
  31. "enabled": provider.enabled,
  32. "config": json.loads(provider.config) if isinstance(provider.config, str) else provider.config,
  33. # Print lifecycle events
  34. "on_print_start": provider.on_print_start,
  35. "on_print_complete": provider.on_print_complete,
  36. "on_print_failed": provider.on_print_failed,
  37. "on_print_stopped": provider.on_print_stopped,
  38. "on_print_progress": provider.on_print_progress,
  39. "on_print_missing_spool_assignment": provider.on_print_missing_spool_assignment,
  40. # Printer status events
  41. "on_printer_offline": provider.on_printer_offline,
  42. "on_printer_error": provider.on_printer_error,
  43. "on_filament_low": provider.on_filament_low,
  44. "on_maintenance_due": provider.on_maintenance_due,
  45. # AMS environmental alarms (regular AMS)
  46. "on_ams_humidity_high": provider.on_ams_humidity_high,
  47. "on_ams_temperature_high": provider.on_ams_temperature_high,
  48. # AMS-HT environmental alarms
  49. "on_ams_ht_humidity_high": provider.on_ams_ht_humidity_high,
  50. "on_ams_ht_temperature_high": provider.on_ams_ht_temperature_high,
  51. # Build plate detection
  52. "on_plate_not_empty": provider.on_plate_not_empty,
  53. # Bed cooled
  54. "on_bed_cooled": provider.on_bed_cooled,
  55. # First layer complete
  56. "on_first_layer_complete": provider.on_first_layer_complete,
  57. # Print queue events
  58. "on_queue_job_added": provider.on_queue_job_added,
  59. "on_queue_job_assigned": provider.on_queue_job_assigned,
  60. "on_queue_job_started": provider.on_queue_job_started,
  61. "on_queue_job_waiting": provider.on_queue_job_waiting,
  62. "on_queue_job_skipped": provider.on_queue_job_skipped,
  63. "on_queue_job_failed": provider.on_queue_job_failed,
  64. "on_queue_completed": provider.on_queue_completed,
  65. # Quiet hours
  66. "quiet_hours_enabled": provider.quiet_hours_enabled,
  67. "quiet_hours_start": provider.quiet_hours_start,
  68. "quiet_hours_end": provider.quiet_hours_end,
  69. # Daily digest
  70. "daily_digest_enabled": provider.daily_digest_enabled,
  71. "daily_digest_time": provider.daily_digest_time,
  72. # Printer filter
  73. "printer_id": provider.printer_id,
  74. # Status tracking
  75. "last_success": provider.last_success,
  76. "last_error": provider.last_error,
  77. "last_error_at": provider.last_error_at,
  78. # Timestamps
  79. "created_at": provider.created_at,
  80. "updated_at": provider.updated_at,
  81. }
  82. # ============================================================================
  83. # Provider List/Create Routes (no path parameters)
  84. # ============================================================================
  85. @router.get("/", response_model=list[NotificationProviderResponse])
  86. async def list_notification_providers(
  87. db: AsyncSession = Depends(get_db),
  88. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_READ),
  89. ):
  90. """List all notification providers."""
  91. result = await db.execute(select(NotificationProvider).order_by(NotificationProvider.created_at.desc()))
  92. providers = result.scalars().all()
  93. return [_provider_to_dict(provider) for provider in providers]
  94. @router.post("/", response_model=NotificationProviderResponse)
  95. async def create_notification_provider(
  96. provider_data: NotificationProviderCreate,
  97. db: AsyncSession = Depends(get_db),
  98. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_CREATE),
  99. ):
  100. """Create a new notification provider."""
  101. provider = NotificationProvider(
  102. name=provider_data.name,
  103. provider_type=provider_data.provider_type.value,
  104. enabled=provider_data.enabled,
  105. config=json.dumps(provider_data.config),
  106. # Print lifecycle events
  107. on_print_start=provider_data.on_print_start,
  108. on_print_complete=provider_data.on_print_complete,
  109. on_print_failed=provider_data.on_print_failed,
  110. on_print_stopped=provider_data.on_print_stopped,
  111. on_print_progress=provider_data.on_print_progress,
  112. on_print_missing_spool_assignment=provider_data.on_print_missing_spool_assignment,
  113. # Printer status events
  114. on_printer_offline=provider_data.on_printer_offline,
  115. on_printer_error=provider_data.on_printer_error,
  116. on_filament_low=provider_data.on_filament_low,
  117. on_maintenance_due=provider_data.on_maintenance_due,
  118. # AMS environmental alarms (regular AMS)
  119. on_ams_humidity_high=provider_data.on_ams_humidity_high,
  120. on_ams_temperature_high=provider_data.on_ams_temperature_high,
  121. # AMS-HT environmental alarms
  122. on_ams_ht_humidity_high=provider_data.on_ams_ht_humidity_high,
  123. on_ams_ht_temperature_high=provider_data.on_ams_ht_temperature_high,
  124. # Build plate detection
  125. on_plate_not_empty=provider_data.on_plate_not_empty,
  126. # Bed cooled
  127. on_bed_cooled=provider_data.on_bed_cooled,
  128. # First layer complete
  129. on_first_layer_complete=provider_data.on_first_layer_complete,
  130. # Print queue events
  131. on_queue_job_added=provider_data.on_queue_job_added,
  132. on_queue_job_assigned=provider_data.on_queue_job_assigned,
  133. on_queue_job_started=provider_data.on_queue_job_started,
  134. on_queue_job_waiting=provider_data.on_queue_job_waiting,
  135. on_queue_job_skipped=provider_data.on_queue_job_skipped,
  136. on_queue_job_failed=provider_data.on_queue_job_failed,
  137. on_queue_completed=provider_data.on_queue_completed,
  138. # Quiet hours
  139. quiet_hours_enabled=provider_data.quiet_hours_enabled,
  140. quiet_hours_start=provider_data.quiet_hours_start,
  141. quiet_hours_end=provider_data.quiet_hours_end,
  142. # Daily digest
  143. daily_digest_enabled=provider_data.daily_digest_enabled,
  144. daily_digest_time=provider_data.daily_digest_time,
  145. # Printer filter
  146. printer_id=provider_data.printer_id,
  147. )
  148. db.add(provider)
  149. await db.commit()
  150. await db.refresh(provider)
  151. logger.info("Created notification provider: %s (%s)", provider.name, provider.provider_type)
  152. return _provider_to_dict(provider)
  153. # ============================================================================
  154. # Static Path Routes (must come BEFORE parameterized routes)
  155. # ============================================================================
  156. @router.post("/test-config", response_model=NotificationTestResponse)
  157. async def test_notification_config(
  158. test_request: NotificationTestRequest,
  159. db: AsyncSession = Depends(get_db),
  160. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_CREATE),
  161. ):
  162. """Test notification configuration before saving."""
  163. success, message = await notification_service.send_test_notification(
  164. test_request.provider_type.value, test_request.config, db
  165. )
  166. return NotificationTestResponse(success=success, message=message)
  167. @router.post("/test-all")
  168. async def test_all_notification_providers(
  169. db: AsyncSession = Depends(get_db),
  170. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_UPDATE),
  171. ):
  172. """Send a test notification to all enabled providers."""
  173. result = await db.execute(select(NotificationProvider).where(NotificationProvider.enabled.is_(True)))
  174. providers = result.scalars().all()
  175. if not providers:
  176. return {"tested": 0, "success": 0, "failed": 0, "results": []}
  177. results = []
  178. success_count = 0
  179. failed_count = 0
  180. for provider in providers:
  181. config = json.loads(provider.config) if isinstance(provider.config, str) else provider.config
  182. success, message = await notification_service.send_test_notification(provider.provider_type, config, db)
  183. # Update provider status
  184. if success:
  185. provider.last_success = datetime.now(timezone.utc)
  186. success_count += 1
  187. else:
  188. provider.last_error = message
  189. provider.last_error_at = datetime.now(timezone.utc)
  190. failed_count += 1
  191. results.append(
  192. {
  193. "provider_id": provider.id,
  194. "provider_name": provider.name,
  195. "provider_type": provider.provider_type,
  196. "success": success,
  197. "message": message,
  198. }
  199. )
  200. await db.commit()
  201. return {
  202. "tested": len(providers),
  203. "success": success_count,
  204. "failed": failed_count,
  205. "results": results,
  206. }
  207. # ============================================================================
  208. # Notification Log Routes (must come BEFORE /{provider_id} routes)
  209. # ============================================================================
  210. @router.get("/logs", response_model=list[NotificationLogResponse])
  211. async def get_notification_logs(
  212. limit: int = Query(default=100, ge=1, le=500),
  213. offset: int = Query(default=0, ge=0),
  214. provider_id: int | None = Query(default=None),
  215. event_type: str | None = Query(default=None),
  216. success: bool | None = Query(default=None),
  217. days: int | None = Query(default=7, ge=1, le=90, description="Filter logs from the last N days"),
  218. db: AsyncSession = Depends(get_db),
  219. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_READ),
  220. ):
  221. """Get notification logs with optional filters."""
  222. query = select(NotificationLog).order_by(desc(NotificationLog.created_at))
  223. # Apply filters
  224. if provider_id is not None:
  225. query = query.where(NotificationLog.provider_id == provider_id)
  226. if event_type is not None:
  227. query = query.where(NotificationLog.event_type == event_type)
  228. if success is not None:
  229. query = query.where(NotificationLog.success == success)
  230. if days is not None:
  231. cutoff = datetime.now(timezone.utc) - timedelta(days=days)
  232. query = query.where(NotificationLog.created_at >= cutoff)
  233. query = query.offset(offset).limit(limit)
  234. result = await db.execute(query)
  235. logs = result.scalars().all()
  236. # Get provider info for each log
  237. response = []
  238. providers_cache: dict[int, NotificationProvider | None] = {}
  239. for log in logs:
  240. if log.provider_id not in providers_cache:
  241. provider_result = await db.execute(
  242. select(NotificationProvider).where(NotificationProvider.id == log.provider_id)
  243. )
  244. providers_cache[log.provider_id] = provider_result.scalar_one_or_none()
  245. provider = providers_cache[log.provider_id]
  246. response.append(
  247. NotificationLogResponse(
  248. id=log.id,
  249. provider_id=log.provider_id,
  250. provider_name=provider.name if provider else None,
  251. provider_type=provider.provider_type if provider else None,
  252. event_type=log.event_type,
  253. title=log.title,
  254. message=log.message,
  255. success=log.success,
  256. error_message=log.error_message,
  257. printer_id=log.printer_id,
  258. printer_name=log.printer_name,
  259. created_at=log.created_at,
  260. )
  261. )
  262. return response
  263. @router.get("/logs/stats", response_model=NotificationLogStats)
  264. async def get_notification_log_stats(
  265. days: int = Query(default=7, ge=1, le=90, description="Statistics for the last N days"),
  266. db: AsyncSession = Depends(get_db),
  267. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_READ),
  268. ):
  269. """Get notification log statistics."""
  270. cutoff = datetime.now(timezone.utc) - timedelta(days=days)
  271. # Total counts
  272. total_result = await db.execute(select(func.count(NotificationLog.id)).where(NotificationLog.created_at >= cutoff))
  273. total = total_result.scalar() or 0
  274. success_result = await db.execute(
  275. select(func.count(NotificationLog.id)).where(
  276. NotificationLog.created_at >= cutoff, NotificationLog.success.is_(True)
  277. )
  278. )
  279. success_count = success_result.scalar() or 0
  280. # By event type
  281. event_result = await db.execute(
  282. select(NotificationLog.event_type, func.count(NotificationLog.id))
  283. .where(NotificationLog.created_at >= cutoff)
  284. .group_by(NotificationLog.event_type)
  285. )
  286. by_event_type = {row[0]: row[1] for row in event_result.fetchall()}
  287. # By provider (need to join to get name)
  288. provider_result = await db.execute(
  289. select(NotificationProvider.name, func.count(NotificationLog.id))
  290. .join(NotificationProvider, NotificationLog.provider_id == NotificationProvider.id)
  291. .where(NotificationLog.created_at >= cutoff)
  292. .group_by(NotificationProvider.name)
  293. )
  294. by_provider = {row[0]: row[1] for row in provider_result.fetchall()}
  295. return NotificationLogStats(
  296. total=total,
  297. success_count=success_count,
  298. failure_count=total - success_count,
  299. by_event_type=by_event_type,
  300. by_provider=by_provider,
  301. )
  302. @router.delete("/logs")
  303. async def clear_notification_logs(
  304. older_than_days: int = Query(default=30, ge=1, description="Delete logs older than N days"),
  305. db: AsyncSession = Depends(get_db),
  306. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_DELETE),
  307. ):
  308. """Clear old notification logs."""
  309. cutoff = datetime.now(timezone.utc) - timedelta(days=older_than_days)
  310. result = await db.execute(delete(NotificationLog).where(NotificationLog.created_at < cutoff))
  311. await db.commit()
  312. deleted_count = result.rowcount
  313. logger.info("Deleted %s notification logs older than %s days", deleted_count, older_than_days)
  314. return {"deleted": deleted_count, "message": f"Deleted {deleted_count} logs older than {older_than_days} days"}
  315. # ============================================================================
  316. # Provider Instance Routes (parameterized - must come LAST)
  317. # ============================================================================
  318. @router.get("/{provider_id}", response_model=NotificationProviderResponse)
  319. async def get_notification_provider(
  320. provider_id: int,
  321. db: AsyncSession = Depends(get_db),
  322. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_READ),
  323. ):
  324. """Get a specific notification provider."""
  325. result = await db.execute(select(NotificationProvider).where(NotificationProvider.id == provider_id))
  326. provider = result.scalar_one_or_none()
  327. if not provider:
  328. raise HTTPException(status_code=404, detail="Notification provider not found")
  329. return _provider_to_dict(provider)
  330. @router.patch("/{provider_id}", response_model=NotificationProviderResponse)
  331. async def update_notification_provider(
  332. provider_id: int,
  333. update_data: NotificationProviderUpdate,
  334. db: AsyncSession = Depends(get_db),
  335. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_UPDATE),
  336. ):
  337. """Update a notification provider."""
  338. result = await db.execute(select(NotificationProvider).where(NotificationProvider.id == provider_id))
  339. provider = result.scalar_one_or_none()
  340. if not provider:
  341. raise HTTPException(status_code=404, detail="Notification provider not found")
  342. # Update only provided fields
  343. update_dict = update_data.model_dump(exclude_unset=True)
  344. for key, value in update_dict.items():
  345. if key == "config" and value is not None:
  346. setattr(provider, key, json.dumps(value))
  347. elif key == "provider_type" and value is not None:
  348. setattr(provider, key, value.value)
  349. else:
  350. setattr(provider, key, value)
  351. await db.commit()
  352. await db.refresh(provider)
  353. logger.info("Updated notification provider: %s", provider.name)
  354. return _provider_to_dict(provider)
  355. @router.delete("/{provider_id}")
  356. async def delete_notification_provider(
  357. provider_id: int,
  358. db: AsyncSession = Depends(get_db),
  359. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_DELETE),
  360. ):
  361. """Delete a notification provider."""
  362. result = await db.execute(select(NotificationProvider).where(NotificationProvider.id == provider_id))
  363. provider = result.scalar_one_or_none()
  364. if not provider:
  365. raise HTTPException(status_code=404, detail="Notification provider not found")
  366. name = provider.name
  367. await db.delete(provider)
  368. await db.commit()
  369. logger.info("Deleted notification provider: %s", name)
  370. return {"message": f"Notification provider '{name}' deleted"}
  371. @router.post("/{provider_id}/test", response_model=NotificationTestResponse)
  372. async def test_notification_provider(
  373. provider_id: int,
  374. db: AsyncSession = Depends(get_db),
  375. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATIONS_UPDATE),
  376. ):
  377. """Send a test notification using an existing provider."""
  378. result = await db.execute(select(NotificationProvider).where(NotificationProvider.id == provider_id))
  379. provider = result.scalar_one_or_none()
  380. if not provider:
  381. raise HTTPException(status_code=404, detail="Notification provider not found")
  382. config = json.loads(provider.config) if isinstance(provider.config, str) else provider.config
  383. success, message = await notification_service.send_test_notification(provider.provider_type, config, db)
  384. # Update provider status
  385. if success:
  386. provider.last_success = datetime.now(timezone.utc)
  387. else:
  388. provider.last_error = message
  389. provider.last_error_at = datetime.now(timezone.utc)
  390. await db.commit()
  391. return NotificationTestResponse(success=success, message=message)