- Add ntfy columns to Settings model (server_url, topic, auth_token, enabled, per-type toggles, lead times) - Create NtfySent dedup model to prevent duplicate notifications - Create ntfy service with SSRF validation and async httpx send - Create ntfy_templates service with per-type payload builders - Create APScheduler background dispatch job (60s interval, events/reminders/todos/projects) - Register scheduler in main.py lifespan with max_instances=1 - Update SettingsUpdate with ntfy validators (URL scheme, topic regex, lead time ranges) - Update SettingsResponse with ntfy fields; ntfy_has_token computed, token never exposed - Add POST /api/settings/ntfy/test endpoint - Update GET/PUT settings to use explicit _to_settings_response() helper - Add Alembic migration 022 for ntfy settings columns + ntfy_sent table - Add httpx==0.27.2 and apscheduler==3.10.4 to requirements.txt Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
25 lines
789 B
Python
25 lines
789 B
Python
from sqlalchemy import String, func
|
|
from sqlalchemy.orm import Mapped, mapped_column
|
|
from datetime import datetime
|
|
from app.database import Base
|
|
|
|
|
|
class NtfySent(Base):
|
|
"""
|
|
Deduplication table for ntfy notifications.
|
|
Prevents the background job from re-sending the same notification
|
|
within a given time window.
|
|
|
|
Key format: "{type}:{entity_id}:{date_window}"
|
|
Examples:
|
|
"reminder:42:2026-02-25"
|
|
"event:17:2026-02-25T09:00"
|
|
"todo:8:2026-02-25"
|
|
"project:3:2026-02-25"
|
|
"""
|
|
__tablename__ = "ntfy_sent"
|
|
|
|
id: Mapped[int] = mapped_column(primary_key=True)
|
|
notification_key: Mapped[str] = mapped_column(String(255), unique=True, index=True)
|
|
sent_at: Mapped[datetime] = mapped_column(default=func.now(), server_default=func.now())
|