667b1a82c3
Per the logo + brand: the name is upbeatBytes (camelCase). Swept all user-facing strings — titles/og:site_name/og:title, logo alt text, share pages (share.py), emails (email_send), classifier prompt (llm), digest/unsubscribe (api), PWA manifest, game share text, sign-in, the SPA shell + patch-static-heads (play title) — plus README/publish.sh and the email test fixture. (SMTP From env was already upbeatBytes.) Domains (upbeatbytes.com) unchanged. 425 BE + 36 FE green. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
125 lines
4.9 KiB
Python
125 lines
4.9 KiB
Python
"""Minimal transactional email over SMTP (magic-link sign-in).
|
|
|
|
Config comes from the environment (GOODNEWS_SMTP_*), supplied to the API
|
|
container via its env_file. STARTTLS submission; plain-text with a simple HTML
|
|
alternative. No third-party email API — just the configured relay.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import os
|
|
import smtplib
|
|
import ssl
|
|
from email.message import EmailMessage
|
|
from html import escape
|
|
|
|
|
|
def smtp_configured() -> bool:
|
|
return bool(os.environ.get("GOODNEWS_SMTP_HOST"))
|
|
|
|
|
|
def _cfg() -> dict:
|
|
return {
|
|
"host": os.environ.get("GOODNEWS_SMTP_HOST", ""),
|
|
"port": int(os.environ.get("GOODNEWS_SMTP_PORT", "587")),
|
|
"user": os.environ.get("GOODNEWS_SMTP_USER", ""),
|
|
"password": os.environ.get("GOODNEWS_SMTP_PASSWORD", ""),
|
|
"sender": os.environ.get("GOODNEWS_SMTP_FROM", "upbeatBytes <hello@upbeatbytes.com>"),
|
|
# Where a reader's reply should land; falls back to the From address.
|
|
"reply_to": os.environ.get("GOODNEWS_REPLY_TO_EMAIL", ""),
|
|
}
|
|
|
|
|
|
def send_email(to: str, subject: str, text: str, html: str | None = None, reply_to: str | None = None,
|
|
headers: dict | None = None) -> None:
|
|
"""Send one message. Raises on failure (caller decides how loud to be)."""
|
|
cfg = _cfg()
|
|
if not cfg["host"]:
|
|
raise RuntimeError("SMTP not configured (set GOODNEWS_SMTP_HOST)")
|
|
msg = EmailMessage()
|
|
msg["From"] = cfg["sender"]
|
|
msg["To"] = to
|
|
if reply_to:
|
|
msg["Reply-To"] = reply_to
|
|
for key, value in (headers or {}).items():
|
|
msg[key] = value
|
|
msg["Subject"] = subject
|
|
msg.set_content(text)
|
|
if html:
|
|
msg.add_alternative(html, subtype="html")
|
|
context = ssl.create_default_context()
|
|
with smtplib.SMTP(cfg["host"], cfg["port"], timeout=20) as server:
|
|
server.ehlo()
|
|
server.starttls(context=context)
|
|
server.ehlo()
|
|
if cfg["user"]:
|
|
server.login(cfg["user"], cfg["password"])
|
|
server.send_message(msg)
|
|
|
|
|
|
def send_feedback(to: str, category: str, message: str, contact: str | None, who: str) -> None:
|
|
"""Notify the admin of new user feedback (plain text is plenty here)."""
|
|
subject = f"upbeatBytes feedback · {category}"
|
|
reply = contact or "(none given)"
|
|
text = (
|
|
f"New feedback ({category})\n"
|
|
f"From: {who}\n"
|
|
f"Reply to: {reply}\n\n"
|
|
f"{message}\n"
|
|
)
|
|
send_email(to, subject, text)
|
|
|
|
|
|
def send_feedback_reply(to: str, reply_text: str, reply_html: str | None, original_message: str) -> None:
|
|
"""Reply to a reader's feedback from the admin inbox. Sends multipart
|
|
text/plain + text/html (the HTML is the pre-sanitized Markdown render). Quotes
|
|
their original note for context; exposes no analytics/account details."""
|
|
subject = "Re: Your upbeatBytes feedback"
|
|
quoted = "\n".join("> " + line for line in (original_message or "").splitlines())
|
|
text = (
|
|
f"{reply_text}\n\n"
|
|
"—\n"
|
|
"In reply to your note to upbeatBytes:\n"
|
|
f"{quoted}\n\n"
|
|
"Thanks for reaching out.\n— upbeatBytes\n"
|
|
)
|
|
body_html = None
|
|
if reply_html:
|
|
oq = escape(original_message or "").replace("\n", "<br>")
|
|
body_html = (
|
|
'<div style="font-family:-apple-system,Segoe UI,Roboto,Helvetica,Arial,sans-serif;'
|
|
'color:#16263a;font-size:15px;line-height:1.5">'
|
|
f"{reply_html}"
|
|
'<p style="color:#5d6b78;font-size:13px;margin-top:20px">In reply to your note to upbeatBytes:</p>'
|
|
f'<blockquote style="color:#5d6b78;border-left:3px solid #e8e3d8;margin:0;padding-left:12px">{oq}</blockquote>'
|
|
"<p>Thanks for reaching out.<br>— upbeatBytes</p></div>"
|
|
)
|
|
# Route the reader's reply to our chosen inbox (never back to the reader).
|
|
cfg = _cfg()
|
|
reply_to = cfg["reply_to"] or cfg["sender"]
|
|
send_email(to, subject, text, html=body_html, reply_to=reply_to)
|
|
|
|
|
|
def send_magic_link(to: str, link: str) -> None:
|
|
"""Send a calm, single-purpose sign-in email."""
|
|
subject = "Your upbeatBytes sign-in link"
|
|
text = (
|
|
"Welcome back to upbeatBytes.\n\n"
|
|
f"Tap to sign in:\n{link}\n\n"
|
|
"This link works once and expires in 15 minutes.\n"
|
|
"If you didn't request it, you can safely ignore this email."
|
|
)
|
|
safe = escape(link, quote=True)
|
|
html = (
|
|
'<div style="font-family:-apple-system,Segoe UI,Roboto,Helvetica,Arial,sans-serif;'
|
|
'color:#16263a;line-height:1.6">'
|
|
"<p>Welcome back to <strong>upbeatBytes</strong>.</p>"
|
|
f'<p><a href="{safe}" style="display:inline-block;background:#0083ad;color:#fff;'
|
|
'text-decoration:none;padding:11px 20px;border-radius:999px;font-weight:600">'
|
|
"Sign in</a></p>"
|
|
'<p style="color:#5d6b78;font-size:14px">This link works once and expires in 15 minutes. '
|
|
"If you didn't request it, you can safely ignore this email.</p>"
|
|
"</div>"
|
|
)
|
|
send_email(to, subject, text, html)
|