|
from flask import Flask, request, render_template, jsonify |
|
import PIL.Image |
|
import google.generativeai as genai |
|
import os |
|
from tempfile import NamedTemporaryFile |
|
|
|
app = Flask(__name__) |
|
|
|
|
|
generation_config = { |
|
"temperature": 1, |
|
"max_output_tokens": 8192, |
|
} |
|
|
|
safety_settings = [ |
|
{ |
|
"category": "HARM_CATEGORY_HARASSMENT", |
|
"threshold": "BLOCK_NONE" |
|
}, |
|
{ |
|
"category": "HARM_CATEGORY_HATE_SPEECH", |
|
"threshold": "BLOCK_NONE" |
|
}, |
|
{ |
|
"category": "HARM_CATEGORY_SEXUALLY_EXPLICIT", |
|
"threshold": "BLOCK_NONE" |
|
}, |
|
{ |
|
"category": "HARM_CATEGORY_DANGEROUS_CONTENT", |
|
"threshold": "BLOCK_NONE" |
|
}, |
|
] |
|
|
|
GOOGLE_API_KEY = os.environ.get("TOKEN") |
|
|
|
genai.configure(api_key=GOOGLE_API_KEY) |
|
|
|
|
|
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** (peut être écrite au brouillon) : Problème (reformulation de la consigne) |
|
* **Pour chaque document :** |
|
* **Présentation (CP1):** Type de document + objet d'étude (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). |
|
* **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). |
|
""", |
|
"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** (peut être écrite au brouillon) : Problème (reformulation de la consigne) |
|
* **Pour chaque document :** |
|
* **Présentation (CP1):** Type de document + objet d'étude (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 = [] |
|
|
|
try: |
|
for image in 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)) |
|
|
|
model = genai.GenerativeModel(model_name="models/gemini-1.5-flash-002", safety_settings=safety_settings) |
|
response = model.generate_content(content, request_options={"timeout": 600}) |
|
|
|
return jsonify({"response": response.text}) |
|
|
|
except Exception as e: |
|
return jsonify({"error": str(e)}), 500 |
|
finally: |
|
for temp_file in temp_files: |
|
try: |
|
os.unlink(temp_file) |
|
except Exception as e: |
|
print(f"Error deleting temporary file {temp_file}: {e}") |