notification_templates.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  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. }
  45. @router.get("", response_model=list[NotificationTemplateResponse])
  46. @router.get("/", response_model=list[NotificationTemplateResponse])
  47. async def get_templates(
  48. db: AsyncSession = Depends(get_db),
  49. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_READ),
  50. ):
  51. """Get all notification templates."""
  52. result = await db.execute(select(NotificationTemplate).order_by(NotificationTemplate.id))
  53. return result.scalars().all()
  54. @router.get("/variables", response_model=list[EventVariablesResponse])
  55. async def get_variables(
  56. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_READ),
  57. ):
  58. """Get available variables for each event type."""
  59. return [
  60. EventVariablesResponse(
  61. event_type=event_type,
  62. event_name=EVENT_NAMES.get(event_type, event_type),
  63. variables=variables,
  64. )
  65. for event_type, variables in EVENT_VARIABLES.items()
  66. ]
  67. @router.get("/{template_id}", response_model=NotificationTemplateResponse)
  68. async def get_template(
  69. template_id: int,
  70. db: AsyncSession = Depends(get_db),
  71. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_READ),
  72. ):
  73. """Get a single notification template."""
  74. result = await db.execute(select(NotificationTemplate).where(NotificationTemplate.id == template_id))
  75. template = result.scalar_one_or_none()
  76. if not template:
  77. raise HTTPException(status_code=404, detail="Template not found")
  78. return template
  79. @router.put("/{template_id}", response_model=NotificationTemplateResponse)
  80. async def update_template(
  81. template_id: int,
  82. update: NotificationTemplateUpdate,
  83. db: AsyncSession = Depends(get_db),
  84. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_UPDATE),
  85. ):
  86. """Update a notification template."""
  87. result = await db.execute(select(NotificationTemplate).where(NotificationTemplate.id == template_id))
  88. template = result.scalar_one_or_none()
  89. if not template:
  90. raise HTTPException(status_code=404, detail="Template not found")
  91. if update.title_template is not None:
  92. template.title_template = update.title_template
  93. if update.body_template is not None:
  94. template.body_template = update.body_template
  95. await db.commit()
  96. await db.refresh(template)
  97. # Clear template cache so changes take effect immediately
  98. notification_service.clear_template_cache()
  99. return template
  100. @router.post("/{template_id}/reset", response_model=NotificationTemplateResponse)
  101. async def reset_template(
  102. template_id: int,
  103. db: AsyncSession = Depends(get_db),
  104. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_UPDATE),
  105. ):
  106. """Reset a notification template to its default values."""
  107. result = await db.execute(select(NotificationTemplate).where(NotificationTemplate.id == template_id))
  108. template = result.scalar_one_or_none()
  109. if not template:
  110. raise HTTPException(status_code=404, detail="Template not found")
  111. # Find the default template
  112. default = next(
  113. (t for t in DEFAULT_TEMPLATES if t["event_type"] == template.event_type),
  114. None,
  115. )
  116. if not default:
  117. raise HTTPException(status_code=500, detail="Default template not found")
  118. template.title_template = default["title_template"]
  119. template.body_template = default["body_template"]
  120. await db.commit()
  121. await db.refresh(template)
  122. # Clear template cache so changes take effect immediately
  123. notification_service.clear_template_cache()
  124. return template
  125. @router.post("/preview", response_model=TemplatePreviewResponse)
  126. async def preview_template(
  127. request: TemplatePreviewRequest,
  128. _: User | None = RequirePermissionIfAuthEnabled(Permission.NOTIFICATION_TEMPLATES_READ),
  129. ):
  130. """Preview a template with sample data."""
  131. sample = SAMPLE_DATA.get(request.event_type, {})
  132. # Safe template rendering - replace missing vars with empty string
  133. def safe_format(template: str, data: dict) -> str:
  134. result = template
  135. for key, value in data.items():
  136. result = result.replace("{" + key + "}", str(value))
  137. # Remove any remaining unreplaced placeholders
  138. import re
  139. result = re.sub(r"\{[a-z_]+\}", "", result)
  140. return result
  141. return TemplatePreviewResponse(
  142. title=safe_format(request.title_template, sample),
  143. body=safe_format(request.body_template, sample),
  144. )