notification_templates.py 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  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.database import get_db
  6. from backend.app.models.notification_template import NotificationTemplate, DEFAULT_TEMPLATES
  7. from backend.app.schemas.notification_template import (
  8. NotificationTemplateResponse,
  9. NotificationTemplateUpdate,
  10. EventVariablesResponse,
  11. TemplatePreviewRequest,
  12. TemplatePreviewResponse,
  13. EVENT_VARIABLES,
  14. SAMPLE_DATA,
  15. )
  16. from backend.app.services.notification_service import notification_service
  17. router = APIRouter(prefix="/notification-templates", tags=["notification-templates"])
  18. # Event type display names
  19. EVENT_NAMES = {
  20. "print_start": "Print Started",
  21. "print_complete": "Print Completed",
  22. "print_failed": "Print Failed",
  23. "print_stopped": "Print Stopped",
  24. "print_progress": "Print Progress",
  25. "printer_offline": "Printer Offline",
  26. "printer_error": "Printer Error",
  27. "filament_low": "Filament Low",
  28. "maintenance_due": "Maintenance Due",
  29. "test": "Test Notification",
  30. }
  31. @router.get("", response_model=list[NotificationTemplateResponse])
  32. async def get_templates(db: AsyncSession = Depends(get_db)):
  33. """Get all notification templates."""
  34. result = await db.execute(
  35. select(NotificationTemplate).order_by(NotificationTemplate.id)
  36. )
  37. return result.scalars().all()
  38. @router.get("/variables", response_model=list[EventVariablesResponse])
  39. async def get_variables():
  40. """Get available variables for each event type."""
  41. return [
  42. EventVariablesResponse(
  43. event_type=event_type,
  44. event_name=EVENT_NAMES.get(event_type, event_type),
  45. variables=variables,
  46. )
  47. for event_type, variables in EVENT_VARIABLES.items()
  48. ]
  49. @router.get("/{template_id}", response_model=NotificationTemplateResponse)
  50. async def get_template(template_id: int, db: AsyncSession = Depends(get_db)):
  51. """Get a single notification template."""
  52. result = await db.execute(
  53. select(NotificationTemplate).where(NotificationTemplate.id == template_id)
  54. )
  55. template = result.scalar_one_or_none()
  56. if not template:
  57. raise HTTPException(status_code=404, detail="Template not found")
  58. return template
  59. @router.put("/{template_id}", response_model=NotificationTemplateResponse)
  60. async def update_template(
  61. template_id: int,
  62. update: NotificationTemplateUpdate,
  63. db: AsyncSession = Depends(get_db),
  64. ):
  65. """Update a notification template."""
  66. result = await db.execute(
  67. select(NotificationTemplate).where(NotificationTemplate.id == template_id)
  68. )
  69. template = result.scalar_one_or_none()
  70. if not template:
  71. raise HTTPException(status_code=404, detail="Template not found")
  72. if update.title_template is not None:
  73. template.title_template = update.title_template
  74. if update.body_template is not None:
  75. template.body_template = update.body_template
  76. await db.commit()
  77. await db.refresh(template)
  78. # Clear template cache so changes take effect immediately
  79. notification_service.clear_template_cache()
  80. return template
  81. @router.post("/{template_id}/reset", response_model=NotificationTemplateResponse)
  82. async def reset_template(template_id: int, db: AsyncSession = Depends(get_db)):
  83. """Reset a notification template to its default values."""
  84. result = await db.execute(
  85. select(NotificationTemplate).where(NotificationTemplate.id == template_id)
  86. )
  87. template = result.scalar_one_or_none()
  88. if not template:
  89. raise HTTPException(status_code=404, detail="Template not found")
  90. # Find the default template
  91. default = next(
  92. (t for t in DEFAULT_TEMPLATES if t["event_type"] == template.event_type),
  93. None,
  94. )
  95. if not default:
  96. raise HTTPException(status_code=500, detail="Default template not found")
  97. template.title_template = default["title_template"]
  98. template.body_template = default["body_template"]
  99. await db.commit()
  100. await db.refresh(template)
  101. # Clear template cache so changes take effect immediately
  102. notification_service.clear_template_cache()
  103. return template
  104. @router.post("/preview", response_model=TemplatePreviewResponse)
  105. async def preview_template(request: TemplatePreviewRequest):
  106. """Preview a template with sample data."""
  107. sample = SAMPLE_DATA.get(request.event_type, {})
  108. # Safe template rendering - replace missing vars with empty string
  109. def safe_format(template: str, data: dict) -> str:
  110. result = template
  111. for key, value in data.items():
  112. result = result.replace("{" + key + "}", str(value))
  113. # Remove any remaining unreplaced placeholders
  114. import re
  115. result = re.sub(r"\{[a-z_]+\}", "", result)
  116. return result
  117. return TemplatePreviewResponse(
  118. title=safe_format(request.title_template, sample),
  119. body=safe_format(request.body_template, sample),
  120. )