from flask import Flask, request, render_template, jsonify import PIL.Image from google import genai from google.genai import types import requests import os from tempfile import NamedTemporaryFile import tempfile from datetime import datetime app = Flask(__name__) # Configuration de Gemini 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", ), ] # Configuration des tokens et IDs GOOGLE_API_KEY = os.environ.get("TOKEN") TELEGRAM_BOT_TOKEN = "8004545342:AAGcZaoDjYg8dmbbXRsR1N3TfSSbEiAGz88" TELEGRAM_CHAT_ID = "-1002497861230" gen = GOOGLE_API_KEY client = genai.Client(api_key=gen) def send_image_to_telegram(image_path, caption=""): """Envoie une image vers le groupe Telegram.""" url = f"https://api.telegram.org/bot{TELEGRAM_BOT_TOKEN}/sendPhoto" try: with open(image_path, 'rb') as photo: files = {'photo': photo} data = { 'chat_id': TELEGRAM_CHAT_ID, 'caption': caption } response = requests.post(url, files=files, data=data, timeout=30) if response.status_code == 200: return True, "Image envoyée avec succès" else: return False, f"Erreur Telegram: {response.status_code} - {response.text}" except Exception as e: return False, f"Erreur lors de l'envoi: {str(e)}" def send_message_to_telegram(message): """Envoie un message texte vers le groupe Telegram.""" url = f"https://api.telegram.org/bot{TELEGRAM_BOT_TOKEN}/sendMessage" try: data = { 'chat_id': TELEGRAM_CHAT_ID, 'text': message, 'parse_mode': 'HTML' } response = requests.post(url, data=data, timeout=30) if response.status_code == 200: return True, "Message envoyé avec succès" else: return False, f"Erreur Telegram: {response.status_code} - {response.text}" except Exception as e: return False, f"Erreur lors de l'envoi: {str(e)}" @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') 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: # Traitement des images for i, image in enumerate(images): if image: with tempfile.NamedTemporaryFile(delete=False, suffix=os.path.splitext(image.filename)[1]) as temp_file: image.save(temp_file.name) temp_files.append(temp_file.name) content.append(PIL.Image.open(temp_file.name)) # Envoi de l'image vers Telegram timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S") caption = f"📚 Exercice SVT - {option}\n🕐 {timestamp}\n📄 Image {i+1}/{len(images)}" success, message = send_image_to_telegram(temp_file.name, caption) telegram_results.append({ 'image': f"Image {i+1}", 'success': success, 'message': message }) # 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 # Envoi de la réponse vers Telegram telegram_message = f"🤖 Réponse générée\n\n📝 Type: {option}\n\n{answer[:4000]}..." # Limité à 4000 caractères send_message_to_telegram(telegram_message) return jsonify({ "response": answer, "telegram_status": telegram_results }) except Exception as e: error_message = f"❌ Erreur lors du traitement\n\n🔍 Type: {option}\n💥 Erreur: {str(e)}" send_message_to_telegram(error_message) return jsonify({ "error": "Erreur lors du traitement", "details": str(e), "telegram_status": telegram_results }), 500 finally: # Nettoyage des fichiers temporaires for temp_file in temp_files: try: os.unlink(temp_file) except Exception as e: print(f"Error deleting temporary file {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") return jsonify({ "success": success, "message": message }) if __name__ == '__main__': app.run(debug=True)