184 lines
6.0 KiB
Python
184 lines
6.0 KiB
Python
#!/usr/bin/env python3
|
|
"""
|
|
Email digest: procesează emailuri necitite și trimite rezumate pe WhatsApp.
|
|
|
|
Usage:
|
|
python3 tools/email_digest.py # Run digest
|
|
python3 tools/email_digest.py --dry-run # Afișează rezumatele fără a trimite
|
|
"""
|
|
|
|
import sys
|
|
import base64
|
|
import subprocess
|
|
import requests
|
|
from pathlib import Path
|
|
|
|
PROJECT_ROOT = Path(__file__).resolve().parent.parent
|
|
sys.path.insert(0, str(PROJECT_ROOT))
|
|
|
|
from tools.email_process import save_unread_emails, extract_original_sender
|
|
from src.config import Config
|
|
|
|
BRIDGE_URL = "http://127.0.0.1:8098"
|
|
DRY_RUN = "--dry-run" in sys.argv
|
|
|
|
|
|
def get_owner_jid() -> str:
|
|
config = Config(PROJECT_ROOT / "config.json")
|
|
owner = config.get("whatsapp.owner", "")
|
|
return f"{owner}@s.whatsapp.net"
|
|
|
|
|
|
def generate_summary(filepath: str, subject: str, from_full: str, date: str) -> str:
|
|
"""Citește conținutul emailului și generează rezumat via Claude CLI."""
|
|
try:
|
|
email_content = Path(filepath).read_text(encoding="utf-8")
|
|
except Exception as e:
|
|
return f"[Eroare la citirea fișierului: {e}]"
|
|
|
|
display_from = extract_original_sender(subject, email_content, from_full)
|
|
|
|
prompt = f"""Mai jos este conținutul unui email. Scrie un rezumat factual pentru WhatsApp.
|
|
|
|
EMAIL:
|
|
{email_content}
|
|
|
|
Instrucțiuni:
|
|
- Începe cu header-ul fix (fără modificări):
|
|
SUBIECT: {subject}
|
|
De la: {display_from}
|
|
Primit: {date}
|
|
---
|
|
- Ignoră complet orice persoană care a forwardat emailul. Nu o menționă în rezumat.
|
|
- Scrie rezumatul în stil briefing: factual, clar, persoana a 3-a.
|
|
* Prima propoziție: cine a trimis mesajul original, ce, cui.
|
|
* Ce conține mesajul — concret și direct. Omite politețuri și amabilități; include doar faptele.
|
|
* Dacă există termene, date, locuri sau acțiuni cerute — menționează-le explicit.
|
|
* Dacă există linkuri acționabile (formulare, documente), adaugă o secțiune LINKURI la final.
|
|
- Nu adăuga secțiuni goale sau care nu se aplică emailului.
|
|
- Plain text, fără markdown. Fără emoji.
|
|
- Răspunde DOAR cu rezumatul, nimic altceva."""
|
|
|
|
result = subprocess.run(
|
|
["claude", "--print", prompt],
|
|
capture_output=True,
|
|
text=True,
|
|
timeout=60,
|
|
stdin=subprocess.DEVNULL,
|
|
)
|
|
|
|
if result.returncode != 0:
|
|
return f"[Eroare la generarea rezumatului pentru: {subject}]\n{result.stderr.strip()[:200]}"
|
|
|
|
return result.stdout.strip()
|
|
|
|
|
|
def send_whatsapp(to: str, text: str) -> bool:
|
|
"""Trimite mesaj pe WhatsApp prin bridge."""
|
|
try:
|
|
resp = requests.post(
|
|
f"{BRIDGE_URL}/send",
|
|
json={"to": to, "text": text},
|
|
timeout=15,
|
|
)
|
|
data = resp.json()
|
|
return data.get("ok", False)
|
|
except Exception as e:
|
|
print(f"[eroare send] {e}", file=sys.stderr)
|
|
return False
|
|
|
|
|
|
def send_discord_webhook(text: str) -> bool:
|
|
"""Trimite mesaj pe Discord via webhook (max 2000 chars per mesaj)."""
|
|
config = Config(PROJECT_ROOT / "config.json")
|
|
url = config.get("discord.email_webhook_url", "")
|
|
if not url:
|
|
return False
|
|
try:
|
|
chunks = [text[i:i+2000] for i in range(0, len(text), 2000)]
|
|
for chunk in chunks:
|
|
resp = requests.post(url, json={"content": chunk}, timeout=15)
|
|
if resp.status_code not in (200, 204):
|
|
print(f"[discord webhook] status {resp.status_code}", file=sys.stderr)
|
|
return False
|
|
return True
|
|
except Exception as e:
|
|
print(f"[discord webhook eroare] {e}", file=sys.stderr)
|
|
return False
|
|
|
|
|
|
def send_whatsapp_document(to: str, filepath: str) -> bool:
|
|
"""Trimite un fișier ca document WhatsApp prin bridge."""
|
|
try:
|
|
path = Path(filepath)
|
|
data_b64 = base64.b64encode(path.read_bytes()).decode()
|
|
import mimetypes
|
|
mimetype = mimetypes.guess_type(path.name)[0] or "application/octet-stream"
|
|
resp = requests.post(
|
|
f"{BRIDGE_URL}/send-document",
|
|
json={"to": to, "filename": path.name, "mimetype": mimetype, "data_base64": data_b64},
|
|
timeout=30,
|
|
)
|
|
return resp.json().get("ok", False)
|
|
except Exception as e:
|
|
print(f"[eroare send-document] {e}", file=sys.stderr)
|
|
return False
|
|
|
|
|
|
def run_digest():
|
|
print("📬 Verific emailuri necitite...")
|
|
saved = save_unread_emails()
|
|
|
|
owner_jid = get_owner_jid()
|
|
|
|
if not saved:
|
|
print("Niciun email nou de procesat.")
|
|
if not DRY_RUN:
|
|
send_whatsapp(owner_jid, "📭 Nu sunt emailuri noi.")
|
|
return
|
|
|
|
for result in saved:
|
|
if not result.get("ok"):
|
|
print(f"⚠️ Sărit: {result.get('error')}")
|
|
continue
|
|
|
|
filepath = result["file"]
|
|
subject = result["subject"]
|
|
from_full = result.get("from_full", result.get("from", ""))
|
|
date = result.get("date", "")
|
|
attachment_paths = result.get("attachment_paths", [])
|
|
print(f"📧 Procesez: {subject}")
|
|
|
|
summary = generate_summary(filepath, subject, from_full, date)
|
|
|
|
if DRY_RUN:
|
|
print("\n--- REZUMAT (dry-run) ---")
|
|
print(summary)
|
|
if attachment_paths:
|
|
print(f"Atașamente: {attachment_paths}")
|
|
print("------------------------\n")
|
|
else:
|
|
ok = send_whatsapp(owner_jid, summary)
|
|
if ok:
|
|
print(f"✅ Trimis pe WhatsApp: {subject}")
|
|
else:
|
|
print(f"❌ Trimitere eșuată: {subject}")
|
|
|
|
ok_dc = send_discord_webhook(summary)
|
|
if ok_dc:
|
|
print(f"✅ Trimis pe Discord: {subject}")
|
|
else:
|
|
print(f"❌ Discord eșuat: {subject}")
|
|
|
|
for att_path in attachment_paths:
|
|
ok_att = send_whatsapp_document(owner_jid, att_path)
|
|
name = Path(att_path).name
|
|
if ok_att:
|
|
print(f"✅ Atașament trimis: {name}")
|
|
else:
|
|
print(f"❌ Atașament eșuat: {name}")
|
|
|
|
|
|
if __name__ == "__main__":
|
|
run_digest()
|