Files
Brancheneinstufung2/fotograf-de-scraper/backend/publish_request_api.py
Floke 1f5805e64c [34588f42] Feat: Versandzeit-Steuerung für Freigabe-Anfragen hinzugefügt
- Backend unterstützt nun zeitgesteuerten Versand (scheduled_time) via BackgroundTasks.
- Frontend um ein Zeitauswahl-Feld erweitert.
2026-04-17 20:21:44 +00:00

168 lines
6.6 KiB
Python

from fastapi import APIRouter, Depends, HTTPException, Request, BackgroundTasks
from pydantic import BaseModel
from sqlalchemy.orm import Session
from database import get_db, DiscountCode
import datetime
import logging
from gmail_service import GmailService
import re
import time
import asyncio
from typing import List, Dict, Optional
router = APIRouter(prefix="/api/publish-request", tags=["publish-request"])
logger = logging.getLogger("publish-request")
class CodesUpload(BaseModel):
codes: str # comma separated
class SendReleaseRequest(BaseModel):
emails: List[Dict[str, str]]
scheduled_time: Optional[str] = None # e.g. "10:00"
async def delayed_send(emails: List[Dict[str, str]], scheduled_time: str, db: Session):
try:
# Calculate delay
now = datetime.datetime.now(datetime.timezone(datetime.timedelta(hours=2))) # Berlin Time Approx
target_h, target_m = map(int, scheduled_time.split(":"))
target_time = now.replace(hour=target_h, minute=target_m, second=0, microsecond=0)
if target_time < now:
target_time += datetime.timedelta(days=1)
delay_seconds = (target_time - now).total_seconds()
logger.info(f"Scheduling {len(emails)} emails for {scheduled_time} (in {delay_seconds} seconds)")
await asyncio.sleep(delay_seconds)
service = GmailService(db)
success_count = 0
for email_data in emails:
if service.send_email(email_data["to"], email_data["subject"], email_data["body"]):
success_count += 1
await asyncio.sleep(1) # Rate limiting
logger.info(f"Scheduled send complete: {success_count}/{len(emails)} success.")
except Exception as e:
logger.exception("Error in delayed_send background task")
@router.post("/send")
async def send_requests(data: SendReleaseRequest, background_tasks: BackgroundTasks, db: Session = Depends(get_db)):
if data.scheduled_time:
background_tasks.add_task(delayed_send, data.emails, data.scheduled_time, db)
return {"status": "scheduled", "message": f"Versand für {data.scheduled_time} geplant."}
# Immediate send
service = GmailService(db)
success = 0
failed = []
for email_data in data.emails:
if service.send_email(email_data["to"], email_data["subject"], email_data["body"]):
success += 1
else:
failed.append(email_data["to"])
return {"status": "success", "success": success, "failed": failed}
@router.get("/stats")
def get_stats(db: Session = Depends(get_db)):
total = db.query(DiscountCode).count()
used = db.query(DiscountCode).filter(DiscountCode.is_used == 1).count()
available = total - used
return {"total": total, "used": used, "available": available}
@router.post("/codes")
def upload_codes(data: CodesUpload, db: Session = Depends(get_db)):
codes_list = [c.strip() for c in data.codes.split(",") if c.strip()]
added = 0
for code in set(codes_list):
existing = db.query(DiscountCode).filter(DiscountCode.code == code).first()
if not existing:
new_code = DiscountCode(code=code, is_used=0)
db.add(new_code)
added += 1
db.commit()
return {"status": "success", "added": added}
class WebhookData(BaseModel):
email: str
@router.post("/webhook")
async def handle_webhook(request: Request, db: Session = Depends(get_db)):
# Try to parse JSON from Google Forms webhook
try:
data = await request.json()
except:
raise HTTPException(status_code=400, detail="Invalid JSON")
# We expect {"email": "..."} or similar from the Google Apps Script
email = data.get("email") or data.get("Email")
if not email:
logger.error(f"Webhook received without email: {data}")
return {"status": "error", "message": "Email not found in webhook payload"}
email = email.strip().lower()
# Check if this email already got a code
already_assigned = db.query(DiscountCode).filter(DiscountCode.assigned_to_email == email).first()
if already_assigned:
logger.info(f"Email {email} already received code {already_assigned.code}")
return {"status": "success", "message": "Already sent"}
# Get a free code
free_code = db.query(DiscountCode).filter(DiscountCode.is_used == 0).first()
if not free_code:
logger.error("NO FREE DISCOUNT CODES LEFT!")
# Fallback logic: Notify admin?
return {"status": "error", "message": "No codes available"}
# Mark as used
free_code.is_used = 1
free_code.assigned_to_email = email
free_code.used_at = datetime.datetime.utcnow()
db.commit()
# Send Thank You Email with GmailService
service = GmailService(db)
subject = "Dankeschön für Eure Freigabe & Euer Rabattcode"
# HTML Signature from frontend
SIGNATURE_HTML = """
<br><br><br>
<div style="font-family: 'Verdana', sans-serif; font-size: 11px; color: #3D3C3F;">
<p><strong>Liebe Grüße,</strong><br>Euer Team von Kinderfotos Erding</p>
<p><strong>Zierl Fotografen GmbH</strong><br>
Anton-Bruckner-Straße 5<br>85435 Erding</p>
<p><a href="https://www.kinderfotos-erding.de" style="color: #3D3C3F; text-decoration: none;">www.kinderfotos-erding.de</a></p>
</div>
"""
body_html = f"""
<p>Hallo,</p>
<p>vielen Dank für Eure Unterstützung und das Ausfüllen der Freigabe!</p>
<p>Als kleines Dankeschön hier Euer 25 € Rabattcode für Eure Bestellung:</p>
<p><strong style="font-size: 18px; color: #4F46E5; padding: 10px; border: 1px dashed #4F46E5; display: inline-block;">{free_code.code}</strong></p>
<p>Bitte wartet mit Eurer Bestellung, falls Ihr noch nicht bestellt habt, bis Ihr diesen Code an der Kasse einlösen könnt.</p>
{SIGNATURE_HTML}
"""
try:
success = service.send_email(email, subject, body_html)
if success:
logger.info(f"Successfully sent code {free_code.code} to {email}")
return {"status": "success", "message": "Email sent"}
else:
logger.error(f"Failed to send email to {email}")
free_code.is_used = 0
free_code.assigned_to_email = None
free_code.used_at = None
db.commit()
return {"status": "error", "message": "Failed to send email"}
except Exception as e:
logger.exception("Error sending webhook email")
free_code.is_used = 0
free_code.assigned_to_email = None
free_code.used_at = None
db.commit()
return {"status": "error", "message": str(e)}