notification_templates.py 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. """API routes for notification template management."""
  2. from fastapi import APIRouter, Depends, HTTPException
  3. from sqlalchemy import select
  4. from sqlalchemy.ext.asyncio import AsyncSession
  5. from backend.app.core.auth import RequirePermissionIfAuthEnabled
  6. from backend.app.core.database import get_db
  7. from backend.app.core.permissions import Permission
  8. from backend.app.models.notification_template import DEFAULT_TEMPLATES, NotificationTemplate
  9. from backend.app.models.user import User
  10. from backend.app.schemas.notification_template import (
  11. EVENT_VARIABLES,
  12. SAMPLE_DATA,
  13. EventVariablesResponse,
  14. NotificationTemplateResponse,
  15. NotificationTemplateUpdate,
  16. TemplatePreviewRequest,
  17. TemplatePreviewResponse,
  18. )
  19. from backend.app.services.notification_service import notification_service
  20. router = APIRouter(prefix="/notification-templates", tags=["notification-templates"])
  21. # Event type display names
  22. EVENT_NAMES = {
  23. "print_start": "Print Started",
  24. "print_complete": "Print Completed",
  25. "print_failed": "Print Failed",
  26. "print_stopped": "Print Stopped",
  27. "print_progress": "Print Progress",
  28. "printer_offline": "Printer Offline",
  29. "printer_error": "Printer Error",
  30. "filament_low": "Filament Low",
  31. "maintenance_due": "Maintenance Due",
  32. "test": "Test Notification",
  33. # Queue notifications
  34. "queue_job_added": "Queue Job Added",
  35. "queue_job_assigned": "Queue Job Assigned",
  36. "queue_job_started": "Queue Job Started",
  37. "queue_job_waiting": "Queue Job Waiting",
  38. "queue_job_skipped": "Queue Job Skipped",
  39. "queue_job_failed": "Queue Job Failed",
  40. "queue_completed": "Queue Completed",
  41. # User management
  42. "user_created": "Welcome Email",
  43. "password_reset": "Password Reset",
  44. # User email print notifications
  45. "user_print_start": "User Print Started Email",
  46. "user_print_complete": "User Print Completed Email",
  47. "user_print_failed": "User Print Failed Email",
  48. "user_print_stopped": "User Print Stopped Email",
  49. }
  50. @router.get("", response_model=list[NotificationTemplateResponse])
  51. @router.get("/", response_model=list[NotificationTemplateResponse])
  52. async def get_templates(
  53. db: AsyncSession = Depends(get_db),
  54. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_READ),
  55. ):
  56. """Get all notification templates."""
  57. result = await db.execute(select(NotificationTemplate).order_by(NotificationTemplate.id))
  58. return result.scalars().all()
  59. @router.get("/variables", response_model=list[EventVariablesResponse])
  60. async def get_variables(
  61. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_READ),
  62. ):
  63. """Get available variables for each event type."""
  64. return [
  65. EventVariablesResponse(
  66. event_type=event_type,
  67. event_name=EVENT_NAMES.get(event_type, event_type),
  68. variables=variables,
  69. )
  70. for event_type, variables in EVENT_VARIABLES.items()
  71. ]
  72. @router.get("/{template_id}", response_model=NotificationTemplateResponse)
  73. async def get_template(
  74. template_id: int,
  75. db: AsyncSession = Depends(get_db),
  76. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_READ),
  77. ):
  78. """Get a single notification template."""
  79. result = await db.execute(select(NotificationTemplate).where(NotificationTemplate.id == template_id))
  80. template = result.scalar_one_or_none()
  81. if not template:
  82. raise HTTPException(status_code=404, detail="Template not found")
  83. return template
  84. @router.put("/{template_id}", response_model=NotificationTemplateResponse)
  85. async def update_template(
  86. template_id: int,
  87. update: NotificationTemplateUpdate,
  88. db: AsyncSession = Depends(get_db),
  89. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_UPDATE),
  90. ):
  91. """Update a notification template."""
  92. result = await db.execute(select(NotificationTemplate).where(NotificationTemplate.id == template_id))
  93. template = result.scalar_one_or_none()
  94. if not template:
  95. raise HTTPException(status_code=404, detail="Template not found")
  96. if update.title_template is not None:
  97. template.title_template = update.title_template
  98. if update.body_template is not None:
  99. template.body_template = update.body_template
  100. await db.commit()
  101. await db.refresh(template)
  102. # Clear template cache so changes take effect immediately
  103. notification_service.clear_template_cache()
  104. return template
  105. @router.post("/{template_id}/reset", response_model=NotificationTemplateResponse)
  106. async def reset_template(
  107. template_id: int,
  108. db: AsyncSession = Depends(get_db),
  109. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_UPDATE),
  110. ):
  111. """Reset a notification template to its default values."""
  112. result = await db.execute(select(NotificationTemplate).where(NotificationTemplate.id == template_id))
  113. template = result.scalar_one_or_none()
  114. if not template:
  115. raise HTTPException(status_code=404, detail="Template not found")
  116. # Find the default template
  117. default = next(
  118. (t for t in DEFAULT_TEMPLATES if t["event_type"] == template.event_type),
  119. None,
  120. )
  121. if not default:
  122. raise HTTPException(status_code=500, detail="Default template not found")
  123. template.title_template = default["title_template"]
  124. template.body_template = default["body_template"]
  125. await db.commit()
  126. await db.refresh(template)
  127. # Clear template cache so changes take effect immediately
  128. notification_service.clear_template_cache()
  129. return template
  130. @router.post("/preview", response_model=TemplatePreviewResponse)
  131. async def preview_template(
  132. request: TemplatePreviewRequest,
  133. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_READ),
  134. ):
  135. """Preview a template with sample data."""
  136. sample = SAMPLE_DATA.get(request.event_type, {})
  137. # Safe template rendering - replace missing vars with empty string
  138. def safe_format(template: str, data: dict) -> str:
  139. result = template
  140. for key, value in data.items():
  141. result = result.replace("{" + key + "}", str(value))
  142. # Remove any remaining unreplaced placeholders
  143. import re
  144. result = re.sub(r"\{[a-z_]+\}", "", result)
  145. return result
  146. return TemplatePreviewResponse(
  147. title=safe_format(request.title_template, sample),
  148. body=safe_format(request.body_template, sample),
  149. )