Reads home/{username}/email_allowlist.json (JSON array of addresses).
Fails safe: if file is missing or address not listed, send is blocked with
an informative message. home/ is gitignored; create the file manually per user.
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
70 lines
2.2 KiB
Python
70 lines
2.2 KiB
Python
"""
|
|
Notification tools — proactively send messages to user channels.
|
|
|
|
nc_talk_send routes through notification.py → channels.json.
|
|
email_send uses the server SMTP config from .env (smtp_server, smtp_from_*).
|
|
"""
|
|
|
|
import asyncio
|
|
import json
|
|
import logging
|
|
|
|
from config import settings
|
|
from persona import get_user
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
def _email_allowlist(username: str) -> list[str]:
|
|
"""Load the per-user email allowlist. Returns empty list if not configured."""
|
|
path = settings.home_root() / username / "email_allowlist.json"
|
|
try:
|
|
return [a.lower().strip() for a in json.loads(path.read_text())]
|
|
except FileNotFoundError:
|
|
return []
|
|
except Exception as e:
|
|
logger.warning("failed to read email_allowlist.json for %s: %s", username, e)
|
|
return []
|
|
|
|
|
|
async def email_send(to: str, subject: str, body: str) -> str:
|
|
"""Send an email via the server's configured SMTP account."""
|
|
username = get_user()
|
|
allowlist = _email_allowlist(username)
|
|
|
|
if not allowlist:
|
|
return (
|
|
"Email blocked — no allowlist configured. "
|
|
f"Add allowed addresses to home/{username}/email_allowlist.json as a JSON array."
|
|
)
|
|
if to.lower().strip() not in allowlist:
|
|
return f"Email blocked — {to} is not in the allowlist for {username}."
|
|
|
|
from email_utils import send_email
|
|
ok = await asyncio.to_thread(
|
|
send_email,
|
|
to_email=to,
|
|
subject=subject,
|
|
body_text=body,
|
|
body_html=body.replace("\n", "<br>"),
|
|
)
|
|
if ok:
|
|
return f"Email sent to {to}."
|
|
return "Failed to send email — check SMTP configuration in .env."
|
|
|
|
|
|
async def nc_talk_send(message: str) -> str:
|
|
"""Send a message to the user via their configured notification channel.
|
|
|
|
Channel is resolved from the user's channels.json (notification_channel key).
|
|
Falls back to Nextcloud Talk if configured. No-op if no channel is set.
|
|
"""
|
|
from notification import notify
|
|
username = get_user()
|
|
try:
|
|
await notify(username, message)
|
|
return f"Message sent to {username}'s notification channel."
|
|
except Exception as e:
|
|
logger.warning("nc_talk_send error for %s: %s", username, e)
|
|
return f"Failed to send notification: {e}"
|