| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357 |
- import os
- import logging
- import discord
- from dotenv import load_dotenv
- from openai import AsyncOpenAI, OpenAIError
- import json
- import urllib3
- # Charger les variables d'environnement depuis le fichier .env
- load_dotenv()
- DISCORD_TOKEN = os.getenv('DISCORD_TOKEN')
- OPENAI_API_KEY = os.getenv('OPENAI_API_KEY')
- DISCORD_CHANNEL_ID = os.getenv('DISCORD_CHANNEL_ID')
- PERSONALITY_PROMPT_FILE = os.getenv('PERSONALITY_PROMPT_FILE', 'personality_prompt.txt')
- CONVERSATION_HISTORY_FILE = os.getenv('CONVERSATION_HISTORY_FILE', 'conversation_history.json')
- CONVERSATION_HISTORY_SIZE = int(os.getenv('CONVERSATION_HISTORY_SIZE', '50'))
- BOT_NAME = os.getenv('BOT_NAME', 'ChatBot')
- MODEL = os.getenv('MODEL', 'gpt-4')
- URL_OPENAI_API = os.getenv('URL_OPENAI_API', 'http://localai.localai.svc.cluster.local:8080/v1')
- TEMPERATURE = float(os.getenv('TEMPERATURE', "1.0"))
- # Initialiser le client OpenAI asynchrone ici
- openai_client = AsyncOpenAI(api_key=OPENAI_API_KEY, base_url=URL_OPENAI_API)
- BOT_VERSION = "2.6.0-penta"
- # Vérifier que les tokens et le prompt de personnalité sont récupérés
- if DISCORD_TOKEN is None or OPENAI_API_KEY is None or DISCORD_CHANNEL_ID is None:
- raise ValueError("Les tokens ou l'ID du canal ne sont pas définis dans les variables d'environnement.")
- if not os.path.isfile(PERSONALITY_PROMPT_FILE):
- raise FileNotFoundError(f"Le fichier de prompt de personnalité '{PERSONALITY_PROMPT_FILE}' est introuvable.")
- # Lire le prompt de personnalité depuis le fichier
- with open(PERSONALITY_PROMPT_FILE, 'r', encoding='utf-8') as f:
- PERSONALITY_PROMPT = f.read().strip()
- # Log configuration
- log_format = '%(asctime)-13s : %(name)-15s : %(levelname)-8s : %(message)s'
- logging.basicConfig(handlers=[logging.FileHandler("./chatbot.log", 'a', 'utf-8')], format=log_format, level="INFO")
- console = logging.StreamHandler()
- console.setLevel(logging.INFO)
- console.setFormatter(logging.Formatter(log_format))
- logger = logging.getLogger(BOT_NAME)
- logger.setLevel("INFO")
- logging.getLogger('').addHandler(console)
- httpx_logger = logging.getLogger('httpx')
- httpx_logger.setLevel(logging.WARNING)
- urllib3.disable_warnings()
- # Initialiser les intents
- intents = discord.Intents.default()
- intents.message_content = True # Activer l'intent pour les contenus de message
- # Liste pour stocker l'historique des conversations
- conversation_history = []
- def filter_message(message):
- """Filtre le contenu d'un retour de modèle de language, comme pour enlever les pensées dans le cas de DeepSeek"""
- if len(message.split('</think>')) > 1:
- result = message.split('</think>')[1]
- elif len(message.split('</response>')) > 1:
- result = message.split('</response>')[1]
- result.rstrip("</s>")
- return result
- def transform_emote(message: str, output: bool) -> str:
- """Remplace les smileys par les codes Discord correspondant"""
- list_emote = [
- (":hap:", "<:hap:355854929073537026>"),
- (":angryvault:", "<:angryvault:585550568806940672>"),
- (":minou:", "<:minou:358054423462936576>"),
- (":cetaitsur:", "<a:cetaitsur:826102032963469324>"),
- (":eh:", "<:eh:395979132896280576>"),
- (":desu:", "<:desu:388007643077410837>"),
- (":bave2:", "<:bave2:412252920558387221>"),
- (":haptriste:", "<:haptriste:358054014262181889>"),
- (":perplexe:", "<:perplexe:358054891274371082>"),
- (":sueur:", "<:sueur:358051940631838721>"),
- (":chien:", "<:chien:507606737646518293>"),
- (":kemar:", "<:kemar:419607012796792842>"),
- (":ouch2:", "<:ouch2:777984650710745138>"),
- (":coeur:", "<:coeur:355853389399195649>"),
- (":what:", "<:what:587019571207077928>")
- ]
- for smiley, discord_code in list_emote:
- if output:
- message = message.replace(smiley, discord_code)
- else:
- message = message.replace(discord_code, smiley)
- return message
- def split_message(message, max_length=2000):
- """Divise un message en plusieurs segments de longueur maximale spécifiée."""
- if len(message) <= max_length:
- return [message]
-
- parts = []
- current_part = ""
-
- for line in message.split('\n'):
- if len(current_part) + len(line) + 1 > max_length:
- parts.append(current_part)
- current_part = line + '\n'
- else:
- current_part += line + '\n'
-
- if current_part:
- parts.append(current_part)
-
- return parts
- def load_conversation_history():
- global conversation_history
- if os.path.isfile(CONVERSATION_HISTORY_FILE):
- try:
- with open(CONVERSATION_HISTORY_FILE, 'r', encoding='utf-8') as f:
- loaded_history = json.load(f)
- # Exclure uniquement le PERSONALITY_PROMPT
- conversation_history = [
- msg for msg in loaded_history
- if not (msg.get("role") == "system" and msg.get("content") == PERSONALITY_PROMPT)
- ]
- logger.info(f"Historique chargé depuis {CONVERSATION_HISTORY_FILE}")
- except Exception as e:
- logger.error(f"Erreur lors du chargement de l'historique : {e}")
- conversation_history = []
- else:
- logger.info(f"Aucun fichier d'historique trouvé. Un nouveau fichier sera créé à {CONVERSATION_HISTORY_FILE}")
- def has_text(text):
- """
- Détermine si le texte fourni est non vide après suppression des espaces.
- """
- return bool(text.strip())
- # Fonction de sauvegarde de l'historique
- def save_conversation_history():
- try:
- with open(CONVERSATION_HISTORY_FILE, 'w', encoding='utf-8') as f:
- json.dump(conversation_history, f, ensure_ascii=False, indent=4)
- except Exception as e:
- logger.error(f"Erreur lors de la sauvegarde de l'historique : {e}")
- # Convertir l'ID du channel en entier
- try:
- chatgpt_channel_id = int(DISCORD_CHANNEL_ID)
- except ValueError:
- raise ValueError("L'ID du channel Discord est invalide. Assurez-vous qu'il s'agit d'un entier.")
- class MyDiscordClient(discord.Client):
- async def close(self):
- global openai_client
- if openai_client is not None:
- await openai_client.close()
- openai_client = None
- await super().close()
- # Initialiser le client Discord avec les intents modifiés
- client_discord = MyDiscordClient(intents=intents)
- # Appeler la fonction pour charger l'historique au démarrage
- load_conversation_history()
- def extract_text_from_message(message):
- content = message.get("content", "")
- if isinstance(content, list):
- # Extraire le texte de chaque élément de la liste
- texts = []
- for part in content:
- if isinstance(part, dict):
- text = part.get("text", "")
- if text:
- texts.append(text)
- return ' '.join(texts)
- elif isinstance(content, str):
- return content
- else:
- return ""
- async def read_text_file(attachment):
- file_bytes = await attachment.read()
- return file_bytes.decode('utf-8')
- async def call_openai_api(user_text, user_name, detail='high'):
- # Préparer le contenu pour l'appel API
- message_to_send = {
- "role": "user",
- "content": [
- {"type": "text", "text": f"{user_name} dit : {transform_emote(user_text, False)}"}
- ]
- }
- # Assembler les messages avec le prompt de personnalité en premier
- messages = [
- {"role": "system", "content": PERSONALITY_PROMPT}
- ] + conversation_history + [message_to_send]
- try:
- response = await openai_client.chat.completions.create(
- model=MODEL,
- messages=messages,
- temperature=TEMPERATURE
- )
- if response:
- reply = response.choices[0].message.content
- await add_to_conversation_history(message_to_send)
- # Ajouter la réponse de l'IA directement à l'historique
- await add_to_conversation_history({
- "role": "assistant",
- "content": reply
- })
- return response
- except Exception as e:
- logger.error(f"Erreur durant l'appel de l'API : {e}")
- return None
- @client_discord.event
- async def on_ready():
- logger.info(f'{BOT_NAME} connecté en tant que {client_discord.user}')
- logger.info(f'Utilisation du modèle {MODEL}')
- if not conversation_history:
- logger.info("Aucun historique trouvé. L'historique commence vide.")
- # Envoyer un message de version dans le canal Discord
- channel = client_discord.get_channel(chatgpt_channel_id)
- if channel:
- try:
- embed = discord.Embed(
- title=f"Bot Démarré",
- description=f"🎉 {BOT_NAME} est en ligne ! Version {BOT_VERSION}\nUtilisation du modèle: **{MODEL}**",
- color=0x2222aa # Bleu
- )
- await channel.send(embed=embed)
- logger.info(f"Message de connexion envoyé dans le canal ID {chatgpt_channel_id}")
- except discord.Forbidden:
- logger.error(f"Permissions insuffisantes pour envoyer des messages dans le canal ID {chatgpt_channel_id}.")
- except discord.HTTPException as e:
- logger.error(f"Erreur lors de l'envoi du message de connexion : {e}")
- else:
- logger.error(f"Canal avec ID {chatgpt_channel_id} non trouvé.")
- @client_discord.event
- async def on_message(message):
- global conversation_history
- # Vérifier si le message provient du canal autorisé
- if message.channel.id != chatgpt_channel_id:
- return
- # Ignorer les messages du bot lui-même
- if message.author == client_discord.user:
- return
- user_text = message.content.strip()
- file_content = None
- attachment_filename = None
- # Vérifier si le message est la commande de réinitialisation
- if user_text.lower() == "!reset_history":
- # Vérifier si l'utilisateur a les permissions administratives
- if not message.author.guild_permissions.administrator:
- await message.channel.send("❌ Vous n'avez pas la permission d'utiliser cette commande.")
- return
- conversation_history = []
- save_conversation_history()
- await message.channel.send("✅ L'historique des conversations a été réinitialisé.")
- logger.info(f"Historique des conversations réinitialisé par {message.author}.")
- return # Arrêter le traitement du message après la réinitialisation
- # Extensions de fichiers autorisées
- allowed_extensions = ['.txt', '.py', '.html', '.css', '.js']
- # Variable pour stocker si le message contient un fichier
- has_file = False
- # Vérifier s'il y a une pièce jointe
- if message.attachments:
- for attachment in message.attachments:
- # Vérifier si c'est un fichier avec une extension autorisée
- if any(attachment.filename.endswith(ext) for ext in allowed_extensions):
- file_content = await read_text_file(attachment)
- attachment_filename = attachment.filename
- break
- # Ajouter le contenu du fichier à la requête si présent
- if file_content:
- user_text += f"\nContenu du fichier {attachment.filename}:\n{file_content}"
- # Vérifier si le texte n'est pas vide après ajout du contenu du fichier
- if not has_text(user_text):
- return # Ne pas appeler l'API si le texte est vide
- async with message.channel.typing():
- try:
- # Appeler l'API OpenAI
- result = await call_openai_api(user_text, message.author.name)
- if result:
- reply = result.choices[0].message.content
- reply = filter_message(reply)
- reply = transform_emote(reply, True)
- message_parts = split_message(reply)
- for part in message_parts:
- await message.channel.send(part)
- # Afficher dans la console
- logging.info(f"Réponse envoyée. ({len(message_parts)} message(s))")
- except Exception as e:
- await message.channel.send("Franchement, je sais pas quoi te répondre. <:haptriste:358054014262181889>")
- logger.error(f"Erreur lors du traitement du texte: {e}")
- async def add_to_conversation_history(new_message):
- global conversation_history
- # Ne pas ajouter le PERSONALITY_PROMPT à l'historique
- if new_message.get("role") == "system" and new_message.get("content") == PERSONALITY_PROMPT:
- logger.debug("PERSONALITY_PROMPT système non ajouté à l'historique.")
- return
- conversation_history.append(new_message)
- save_conversation_history()
- logger.debug(f"Message ajouté à l'historique. Taille actuelle : {len(conversation_history)}")
- if len(conversation_history) > CONVERSATION_HISTORY_SIZE:
- logger.info(f"Limite de {CONVERSATION_HISTORY_SIZE} messages atteinte.")
- excess_messages = len(conversation_history) - CONVERSATION_HISTORY_SIZE
- if excess_messages > 0:
- # Supprimer les messages les plus anciens
- del conversation_history[:excess_messages]
- save_conversation_history()
- logger.info(f"{excess_messages} messages les plus anciens ont été supprimés pour maintenir l'historique à {CONVERSATION_HISTORY_SIZE} messages.")
- # Démarrer le bot Discord
- client_discord.run(DISCORD_TOKEN)
|