notification_templates.py 6.1 KB

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