notification_templates.py 4.9 KB

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