|
from flask import Flask, request, render_template, jsonify |
|
import PIL.Image |
|
from google import genai |
|
from google.genai import types |
|
import telebot |
|
import os |
|
from tempfile import NamedTemporaryFile |
|
import tempfile |
|
from datetime import datetime |
|
import logging |
|
import time |
|
|
|
app = Flask(__name__) |
|
|
|
|
|
logging.basicConfig(level=logging.DEBUG) |
|
logger = logging.getLogger(__name__) |
|
|
|
|
|
generation_config = { |
|
"temperature": 1, |
|
"max_output_tokens": 8192, |
|
} |
|
|
|
safety_settings = [ |
|
types.SafetySetting( |
|
category="HARM_CATEGORY_HARASSMENT", |
|
threshold="BLOCK_NONE", |
|
), |
|
types.SafetySetting( |
|
category="HARM_CATEGORY_HATE_SPEECH", |
|
threshold="BLOCK_NONE", |
|
), |
|
types.SafetySetting( |
|
category="HARM_CATEGORY_SEXUALLY_EXPLICIT", |
|
threshold="BLOCK_NONE", |
|
), |
|
types.SafetySetting( |
|
category="HARM_CATEGORY_DANGEROUS_CONTENT", |
|
threshold="BLOCK_NONE", |
|
), |
|
] |
|
|
|
|
|
GOOGLE_API_KEY = os.environ.get("TOKEN") |
|
TELEGRAM_BOT_TOKEN = "8180304240:AAGJZ_MJ6eKtbymxkqzjgOJCr6PWb7uas9U" |
|
TELEGRAM_CHAT_ID = "-4972732072" |
|
|
|
gen = GOOGLE_API_KEY |
|
client = genai.Client(api_key=gen) |
|
|
|
|
|
try: |
|
bot = telebot.TeleBot( |
|
TELEGRAM_BOT_TOKEN, |
|
parse_mode='HTML', |
|
disable_web_page_preview=True, |
|
timeout=60, |
|
num_threads=2 |
|
) |
|
logger.info("Bot Telegram initialisé avec succès") |
|
except Exception as e: |
|
logger.error(f"Erreur initialisation bot: {e}") |
|
bot = None |
|
|
|
def send_image_to_telegram(image_path, caption="", max_retries=3): |
|
"""Envoie une image vers le groupe Telegram avec pyTelegramBotAPI.""" |
|
if not bot: |
|
return False, "Bot Telegram non initialisé" |
|
|
|
try: |
|
|
|
if not os.path.exists(image_path): |
|
logger.error(f"Le fichier {image_path} n'existe pas") |
|
return False, f"Le fichier {image_path} n'existe pas" |
|
|
|
file_size = os.path.getsize(image_path) |
|
if file_size > 50 * 1024 * 1024: |
|
logger.error(f"Fichier trop volumineux: {file_size} bytes") |
|
return False, f"Fichier trop volumineux: {file_size} bytes (limite: 50MB)" |
|
|
|
if file_size == 0: |
|
logger.error(f"Fichier vide: {image_path}") |
|
return False, f"Fichier vide: {image_path}" |
|
|
|
logger.info(f"Envoi de l'image: {image_path} (taille: {file_size} bytes)") |
|
|
|
|
|
for attempt in range(max_retries): |
|
try: |
|
logger.info(f"Tentative {attempt + 1}/{max_retries}") |
|
|
|
with open(image_path, 'rb') as photo: |
|
|
|
if file_size < 10 * 1024 * 1024: |
|
message = bot.send_photo( |
|
chat_id=TELEGRAM_CHAT_ID, |
|
photo=photo, |
|
caption=caption[:1024] if caption else "", |
|
timeout=120 |
|
) |
|
else: |
|
|
|
message = bot.send_document( |
|
chat_id=TELEGRAM_CHAT_ID, |
|
document=photo, |
|
caption=caption[:1024] if caption else "", |
|
timeout=120 |
|
) |
|
|
|
logger.info(f"✅ Image envoyée avec succès (Message ID: {message.message_id})") |
|
return True, f"Image envoyée avec succès (tentative {attempt + 1})" |
|
|
|
except telebot.apihelper.ApiTelegramException as e: |
|
logger.error(f"Erreur API Telegram (tentative {attempt + 1}): {e}") |
|
if attempt == max_retries - 1: |
|
return False, f"Erreur API Telegram après {max_retries} tentatives: {str(e)}" |
|
time.sleep(2 ** attempt) |
|
|
|
except Exception as e: |
|
logger.error(f"Erreur inattendue (tentative {attempt + 1}): {e}") |
|
if attempt == max_retries - 1: |
|
return False, f"Erreur inattendue après {max_retries} tentatives: {str(e)}" |
|
time.sleep(2 ** attempt) |
|
|
|
return False, f"Échec après {max_retries} tentatives" |
|
|
|
except Exception as e: |
|
logger.error(f"Erreur critique lors de l'envoi: {str(e)}") |
|
return False, f"Erreur critique: {str(e)}" |
|
|
|
def send_message_to_telegram(message, max_retries=3): |
|
"""Envoie un message texte vers le groupe Telegram.""" |
|
if not bot: |
|
return False, "Bot Telegram non initialisé" |
|
|
|
try: |
|
|
|
max_length = 4000 |
|
messages = [] |
|
|
|
if len(message) <= max_length: |
|
messages = [message] |
|
else: |
|
|
|
for i in range(0, len(message), max_length): |
|
chunk = message[i:i + max_length] |
|
messages.append(chunk) |
|
|
|
|
|
for i, msg_part in enumerate(messages): |
|
for attempt in range(max_retries): |
|
try: |
|
logger.info(f"Envoi message partie {i + 1}/{len(messages)} (tentative {attempt + 1})") |
|
|
|
sent_message = bot.send_message( |
|
chat_id=TELEGRAM_CHAT_ID, |
|
text=msg_part, |
|
parse_mode='HTML', |
|
timeout=60 |
|
) |
|
|
|
logger.info(f"✅ Message partie {i + 1} envoyé (ID: {sent_message.message_id})") |
|
break |
|
|
|
except telebot.apihelper.ApiTelegramException as e: |
|
logger.error(f"Erreur API Telegram partie {i + 1} (tentative {attempt + 1}): {e}") |
|
if attempt == max_retries - 1: |
|
return False, f"Erreur API après {max_retries} tentatives: {str(e)}" |
|
time.sleep(2 ** attempt) |
|
|
|
except Exception as e: |
|
logger.error(f"Erreur inattendue partie {i + 1} (tentative {attempt + 1}): {e}") |
|
if attempt == max_retries - 1: |
|
return False, f"Erreur inattendue après {max_retries} tentatives: {str(e)}" |
|
time.sleep(2 ** attempt) |
|
|
|
return True, f"Message envoyé avec succès ({len(messages)} partie(s))" |
|
|
|
except Exception as e: |
|
logger.error(f"Erreur critique envoi message: {str(e)}") |
|
return False, f"Erreur critique: {str(e)}" |
|
|
|
def test_bot_permissions(): |
|
"""Teste les permissions du bot dans le groupe.""" |
|
if not bot: |
|
return False, "Bot Telegram non initialisé" |
|
|
|
try: |
|
logger.info("Test des permissions du bot...") |
|
|
|
|
|
chat = bot.get_chat(TELEGRAM_CHAT_ID) |
|
logger.info(f"Chat trouvé: {chat.title if hasattr(chat, 'title') else 'Chat privé'}") |
|
|
|
|
|
bot_info = bot.get_me() |
|
logger.info(f"Bot: @{bot_info.username} ({bot_info.first_name})") |
|
|
|
|
|
test_message = bot.send_message( |
|
chat_id=TELEGRAM_CHAT_ID, |
|
text="🔧 Test de permissions réussi", |
|
timeout=30 |
|
) |
|
|
|
return True, f"Bot @{bot_info.username} autorisé dans '{chat.title if hasattr(chat, 'title') else 'le chat'}'" |
|
|
|
except telebot.apihelper.ApiTelegramException as e: |
|
logger.error(f"Erreur API lors du test: {e}") |
|
return False, f"Erreur API Telegram: {str(e)}" |
|
except Exception as e: |
|
logger.error(f"Erreur lors du test des permissions: {str(e)}") |
|
return False, f"Erreur: {str(e)}" |
|
|
|
def get_bot_status(): |
|
"""Obtient le statut du bot et des connexions.""" |
|
status = { |
|
"bot_initialized": bot is not None, |
|
"bot_info": None, |
|
"chat_info": None, |
|
"api_accessible": False |
|
} |
|
|
|
if bot: |
|
try: |
|
|
|
bot_info = bot.get_me() |
|
status["bot_info"] = { |
|
"username": bot_info.username, |
|
"first_name": bot_info.first_name, |
|
"id": bot_info.id, |
|
"is_bot": bot_info.is_bot |
|
} |
|
status["api_accessible"] = True |
|
|
|
|
|
try: |
|
chat = bot.get_chat(TELEGRAM_CHAT_ID) |
|
status["chat_info"] = { |
|
"id": chat.id, |
|
"type": chat.type, |
|
"title": getattr(chat, 'title', 'N/A'), |
|
"member_count": getattr(chat, 'member_count', 'N/A') |
|
} |
|
except Exception as e: |
|
status["chat_error"] = str(e) |
|
|
|
except Exception as e: |
|
status["bot_error"] = str(e) |
|
|
|
return status |
|
|
|
|
|
@app.route('/bot_status', methods=['GET']) |
|
def bot_status(): |
|
"""Route pour obtenir le statut du bot.""" |
|
return jsonify(get_bot_status()) |
|
|
|
@app.route('/test_bot_permissions', methods=['GET']) |
|
def test_bot_permissions_route(): |
|
"""Route pour tester les permissions du bot.""" |
|
success, message = test_bot_permissions() |
|
return jsonify({ |
|
"success": success, |
|
"message": message, |
|
"bot_status": get_bot_status() |
|
}) |
|
|
|
@app.route('/', methods=['GET']) |
|
def svt(): |
|
"""Renders the SVT page.""" |
|
return render_template("svt.html") |
|
|
|
methodologie_svt = { |
|
"Restitution organisée des connaissances": """ |
|
**Restitution organisée des connaissances (ROC)** |
|
|
|
**Objectif:** Exposer, dans un texte structuré, scientifiquement et grammaticalement correct, illustré si nécessaire, des connaissances sur un point du programme. |
|
|
|
**Structure de la ROC:** |
|
|
|
* **Introduction:** |
|
* Contexte: Synthèse des savoirs (prérequis) nécessaires pour aborder le thème. |
|
* Problème: Reformulation de la consigne sous forme interrogative, découlant logiquement du contexte. |
|
* Plan: Annonce des parties du développement. |
|
* **Développement:** |
|
* Au moins deux paragraphes séparés par une ligne, débutant par un titre souligné. |
|
* Titres: Reprise des parties annoncées dans le plan. |
|
* Contenu: Solution du problème, articulation logique des paragraphes. |
|
* Schéma: Si la consigne l'exige. |
|
* **Conclusion:** |
|
* Réponse logique au problème posé dans l'introduction. |
|
* Intégration des aspects développés. |
|
* Correspondance à la thématique de l'exercice. |
|
|
|
**Conseils:** |
|
|
|
* L'exercice ne comporte pas de documents. |
|
* Le sujet comporte un thème, un contexte et une consigne. |
|
* L'exercice est pondéré sur 7 à 8 points. |
|
""", |
|
"Exploitation du document": """ |
|
**Exploitation de documents (ED)** |
|
|
|
**Objectif:** Trouver le lien entre les informations présentées par un des documents et les connaissances d'un segment de connaissances (partie du programme) en vue de la résolution d'un problème scientifique. |
|
|
|
**Structure de l'ED:** |
|
|
|
* **Introduction** (écrite au brouillon) : Problème (reformulation de la consigne) |
|
* **Pour chaque document :** |
|
* **Présentation (CP1):** Type de document + titre (cf. titre). |
|
* **Analyse (CP2):** Description du fait expérimental (comparaison de courbes, résultats d'expérience) et/ou présentation du fait d'observation (changement de coloration, % de phénotypes). détaille bien cette partie |
|
* **Information saisie (CP3):** Conclusion partielle, fait à interpréter. |
|
* **Mise en relation (CP4):** Interprétation de l'information saisie en utilisant les connaissances acquises, signification permettant la résolution du problème. |
|
* **Synthèse des mises en relation (CP5)** : Lien pertinent et cohérent entre toutes les significations (mise en relation) des informations utiles pour résoudre le problème (répondre à la consigne). Cette partie est séparée du reste par deux lignes. |
|
|
|
**Conseils:** |
|
|
|
* L'exercice comporte un thème, un contexte, une consigne, un ou deux documents, et une pondération (7-8 points). |
|
* Le contexte établit le lien entre le thème et les documents. |
|
* La consigne guide l'élève dans les différentes tâches. |
|
* Les documents doivent comporter un titre et une source, être pertinents, lisibles, et suivre l'ordre chronologique de la résolution. |
|
* Ne pas paraphraser, copier ou faire une description intégrale dans l'analyse (CP2). |
|
* La tâche 4 (CP4) est spécifique à cet exercice. |
|
* Mentionner le document traité (ex: Document 1). |
|
|
|
réponse attendu uniquement en langue française. |
|
""", |
|
"Synthèse": """ |
|
**Élaboration d'une synthèse (ES)** |
|
|
|
**Objectif:** Dégager des informations pertinentes d'un ensemble de documents en vue de résoudre un problème scientifique. La résolution du problème ne fait pas appel directement aux connaissances du cours. |
|
|
|
**Structure de l'ES:** |
|
|
|
* **Introduction** (écrite au brouillon) : Problème (reformulation de la consigne) |
|
* **Pour chaque document :** |
|
* **Présentation (CP1):** Type de document + titre (cf. titre). |
|
* **Analyse (CP2):** Description du fait expérimental (comparaison de courbes, résultats d'expérience) et/ou présentation du fait d'observation (changement de coloration, % de phénotypes). |
|
* **Conclusion partielle (CP3):** Synthèse de l'analyse, élément de réponse au problème. |
|
* **Conclusion générale (CP4):** Récapitulation des conclusions partielles, réponse à la consigne. |
|
|
|
**Conseils:** |
|
|
|
* L'exercice comporte un thème, un contexte, une consigne, deux ou trois documents, et une pondération (5 points). |
|
* Le contexte établit le lien entre le thème et les documents. |
|
* La consigne guide l'élève dans les différentes tâches. |
|
* Les documents doivent comporter un titre et une source, être pertinents, lisibles, et suivre l'ordre chronologique de la résolution. |
|
* Ne pas paraphraser, copier ou faire une description intégrale dans l'analyse (CP2). |
|
* La tâche 3 (CP3) est spécifique à cet exercice. |
|
* Mentionner le document traité (ex: Document 1). |
|
* Toutes les informations nécessaires sont dans les documents. |
|
""" |
|
} |
|
|
|
@app.route('/svt_submit', methods=['POST']) |
|
def svt_submit(): |
|
"""Handles the submission of SVT exercises.""" |
|
option = request.form.get('option') |
|
images = request.files.getlist('images') |
|
|
|
logger.info(f"Traitement d'un exercice SVT: {option}") |
|
logger.info(f"Nombre d'images reçues: {len(images)}") |
|
|
|
content = [f"J'aimerais que tu traites entièrement cet exercice en respectant scrupuleusement la méthodologie d'SVT suivante :\n\n{methodologie_svt[option]}\n\nLe type d'exercice selon la méthodologie est : {option}. Voici les images de l'exercice:"] |
|
temp_files = [] |
|
telegram_results = [] |
|
|
|
try: |
|
|
|
bot_status = get_bot_status() |
|
logger.info(f"Statut du bot: {bot_status}") |
|
|
|
|
|
for i, image in enumerate(images): |
|
if image and image.filename: |
|
logger.info(f"Traitement de l'image {i+1}: {image.filename}") |
|
|
|
|
|
file_extension = os.path.splitext(image.filename)[1].lower() |
|
if not file_extension: |
|
file_extension = '.jpg' |
|
|
|
with tempfile.NamedTemporaryFile(delete=False, suffix=file_extension) as temp_file: |
|
|
|
image.save(temp_file.name) |
|
temp_files.append(temp_file.name) |
|
|
|
|
|
if os.path.getsize(temp_file.name) > 0: |
|
logger.info(f"Image sauvegardée: {temp_file.name} ({os.path.getsize(temp_file.name)} bytes)") |
|
|
|
|
|
try: |
|
pil_image = PIL.Image.open(temp_file.name) |
|
content.append(pil_image) |
|
logger.info(f"Image ajoutée au contenu Gemini: {pil_image.size}") |
|
except Exception as e: |
|
logger.error(f"Erreur lors de l'ouverture de l'image pour Gemini: {e}") |
|
|
|
|
|
timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S") |
|
caption = f"📚 Exercice SVT - {option}\n🕐 {timestamp}\n📄 Image {i+1}/{len([img for img in images if img.filename])}" |
|
|
|
logger.info(f"Tentative d'envoi vers Telegram: {temp_file.name}") |
|
success, message = send_image_to_telegram(temp_file.name, caption) |
|
telegram_results.append({ |
|
'image': f"Image {i+1} - {image.filename}", |
|
'success': success, |
|
'message': message |
|
}) |
|
|
|
if success: |
|
logger.info(f"✅ Image {i+1} envoyée avec succès") |
|
else: |
|
logger.error(f"❌ Échec envoi image {i+1}: {message}") |
|
else: |
|
logger.error(f"Fichier temporaire vide pour l'image {i+1}") |
|
telegram_results.append({ |
|
'image': f"Image {i+1} - {image.filename}", |
|
'success': False, |
|
'message': "Fichier temporaire vide" |
|
}) |
|
|
|
logger.info(f"Résultats Telegram: {telegram_results}") |
|
|
|
|
|
logger.info("Génération de la réponse avec Gemini...") |
|
response = client.models.generate_content( |
|
model="gemini-2.5-flash", |
|
contents=[content], |
|
config=types.GenerateContentConfig( |
|
safety_settings=safety_settings, |
|
) |
|
) |
|
|
|
answer = response.candidates[0].content.parts[0].text |
|
logger.info("Réponse générée avec succès") |
|
|
|
|
|
telegram_message = f"🤖 <b>Réponse générée</b>\n\n📝 <b>Type:</b> {option}\n\n{answer}" |
|
msg_success, msg_message = send_message_to_telegram(telegram_message) |
|
logger.info(f"Envoi réponse Telegram: {msg_success} - {msg_message}") |
|
|
|
return jsonify({ |
|
"response": answer, |
|
"telegram_status": telegram_results, |
|
"bot_status": bot_status, |
|
"response_sent": {"success": msg_success, "message": msg_message} |
|
}) |
|
|
|
except Exception as e: |
|
logger.error(f"Erreur lors du traitement: {str(e)}") |
|
error_message = f"❌ <b>Erreur lors du traitement</b>\n\n🔍 <b>Type:</b> {option}\n💥 <b>Erreur:</b> {str(e)}" |
|
send_message_to_telegram(error_message) |
|
|
|
return jsonify({ |
|
"error": "Erreur lors du traitement", |
|
"details": str(e), |
|
"telegram_status": telegram_results, |
|
"bot_status": get_bot_status() |
|
}), 500 |
|
|
|
finally: |
|
|
|
for temp_file in temp_files: |
|
try: |
|
if os.path.exists(temp_file): |
|
os.unlink(temp_file) |
|
logger.info(f"Fichier temporaire supprimé: {temp_file}") |
|
except Exception as e: |
|
logger.error(f"Erreur lors de la suppression du fichier {temp_file}: {e}") |
|
|
|
@app.route('/test_telegram', methods=['GET']) |
|
def test_telegram(): |
|
"""Route pour tester la connexion Telegram.""" |
|
success, message = send_message_to_telegram("🧪 Test de connexion du bot SVT avec pyTelegramBotAPI") |
|
return jsonify({ |
|
"success": success, |
|
"message": message, |
|
"bot_status": get_bot_status() |
|
}) |
|
|
|
@app.route('/debug_telegram', methods=['GET']) |
|
def debug_telegram(): |
|
"""Route pour débugger les problèmes Telegram.""" |
|
results = {} |
|
|
|
|
|
results['bot_status'] = get_bot_status() |
|
|
|
|
|
perm_success, perm_message = test_bot_permissions() |
|
results['permissions'] = {"success": perm_success, "message": perm_message} |
|
|
|
|
|
msg_success, msg_message = send_message_to_telegram("🔧 Test de débogage avec pyTelegramBotAPI") |
|
results['message_test'] = {"success": msg_success, "message": msg_message} |
|
|
|
|
|
results['config'] = { |
|
"bot_token_set": bool(TELEGRAM_BOT_TOKEN), |
|
"chat_id_set": bool(TELEGRAM_CHAT_ID), |
|
"bot_token_format": TELEGRAM_BOT_TOKEN.startswith("8180304240:") if TELEGRAM_BOT_TOKEN else False, |
|
"chat_id_format": TELEGRAM_CHAT_ID.startswith("-") if TELEGRAM_CHAT_ID else False |
|
} |
|
|
|
return jsonify(results) |
|
|
|
if __name__ == '__main__': |
|
app.run(debug=True) |