notification_templates.py 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  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 DEFAULT_TEMPLATES, NotificationTemplate
  7. from backend.app.schemas.notification_template import (
  8. EVENT_VARIABLES,
  9. SAMPLE_DATA,
  10. EventVariablesResponse,
  11. NotificationTemplateResponse,
  12. NotificationTemplateUpdate,
  13. TemplatePreviewRequest,
  14. TemplatePreviewResponse,
  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. @router.get("/", response_model=list[NotificationTemplateResponse])
  33. async def get_templates(db: AsyncSession = Depends(get_db)):
  34. """Get all notification templates."""
  35. result = await db.execute(select(NotificationTemplate).order_by(NotificationTemplate.id))
  36. return result.scalars().all()
  37. @router.get("/variables", response_model=list[EventVariablesResponse])
  38. async def get_variables():
  39. """Get available variables for each event type."""
  40. return [
  41. EventVariablesResponse(
  42. event_type=event_type,
  43. event_name=EVENT_NAMES.get(event_type, event_type),
  44. variables=variables,
  45. )
  46. for event_type, variables in EVENT_VARIABLES.items()
  47. ]
  48. @router.get("/{template_id}", response_model=NotificationTemplateResponse)
  49. async def get_template(template_id: int, db: AsyncSession = Depends(get_db)):
  50. """Get a single notification template."""
  51. result = await db.execute(select(NotificationTemplate).where(NotificationTemplate.id == template_id))
  52. template = result.scalar_one_or_none()
  53. if not template:
  54. raise HTTPException(status_code=404, detail="Template not found")
  55. return template
  56. @router.put("/{template_id}", response_model=NotificationTemplateResponse)
  57. async def update_template(
  58. template_id: int,
  59. update: NotificationTemplateUpdate,
  60. db: AsyncSession = Depends(get_db),
  61. ):
  62. """Update a notification template."""
  63. result = await db.execute(select(NotificationTemplate).where(NotificationTemplate.id == template_id))
  64. template = result.scalar_one_or_none()
  65. if not template:
  66. raise HTTPException(status_code=404, detail="Template not found")
  67. if update.title_template is not None:
  68. template.title_template = update.title_template
  69. if update.body_template is not None:
  70. template.body_template = update.body_template
  71. await db.commit()
  72. await db.refresh(template)
  73. # Clear template cache so changes take effect immediately
  74. notification_service.clear_template_cache()
  75. return template
  76. @router.post("/{template_id}/reset", response_model=NotificationTemplateResponse)
  77. async def reset_template(template_id: int, db: AsyncSession = Depends(get_db)):
  78. """Reset a notification template to its default values."""
  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. # Find the default template
  84. default = next(
  85. (t for t in DEFAULT_TEMPLATES if t["event_type"] == template.event_type),
  86. None,
  87. )
  88. if not default:
  89. raise HTTPException(status_code=500, detail="Default template not found")
  90. template.title_template = default["title_template"]
  91. template.body_template = default["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("/preview", response_model=TemplatePreviewResponse)
  98. async def preview_template(request: TemplatePreviewRequest):
  99. """Preview a template with sample data."""
  100. sample = SAMPLE_DATA.get(request.event_type, {})
  101. # Safe template rendering - replace missing vars with empty string
  102. def safe_format(template: str, data: dict) -> str:
  103. result = template
  104. for key, value in data.items():
  105. result = result.replace("{" + key + "}", str(value))
  106. # Remove any remaining unreplaced placeholders
  107. import re
  108. result = re.sub(r"\{[a-z_]+\}", "", result)
  109. return result
  110. return TemplatePreviewResponse(
  111. title=safe_format(request.title_template, sample),
  112. body=safe_format(request.body_template, sample),
  113. )