chatbot.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575
  1. import os
  2. import logging
  3. import discord
  4. import json
  5. import urllib3
  6. import base64
  7. import re
  8. from dotenv import load_dotenv
  9. from openai import AsyncOpenAI, OpenAIError
  10. from PIL import Image
  11. from io import BytesIO
  12. from datetime import datetime, timezone, timedelta
  13. from charset_normalizer import from_bytes
  14. from discord.ext import tasks
  15. # Charger les variables d'environnement depuis le fichier .env
  16. load_dotenv()
  17. DISCORD_TOKEN = os.getenv('DISCORD_TOKEN')
  18. OPENAI_API_KEY = os.getenv('OPENAI_API_KEY')
  19. DISCORD_CHANNEL_ID = os.getenv('DISCORD_CHANNEL_ID')
  20. PERSONALITY_PROMPT_FILE = os.getenv('PERSONALITY_PROMPT_FILE', 'personality_prompt.txt')
  21. CONVERSATION_HISTORY_FILE = os.getenv('CONVERSATION_HISTORY_FILE', 'conversation_history.json')
  22. CONVERSATION_HISTORY_SIZE = int(os.getenv('CONVERSATION_HISTORY_SIZE', '100'))
  23. BOT_NAME = os.getenv('BOT_NAME', 'ChatBot')
  24. MODEL = os.getenv('MODEL', 'llama:3.2')
  25. URL_OPENAI_API = os.getenv('URL_OPENAI_API', 'http://openwebui:8080/v1')
  26. TEMPERATURE = float(os.getenv('TEMPERATURE', "1.0"))
  27. BOOT_MESSAGE = os.getenv('BOOT_MESSAGE', "true").lower()
  28. LOG_LEVEL = os.getenv('LOG_LEVEL', "INFO").upper()
  29. HISTORY_ANALYSIS_IMAGE = os.getenv('HISTORY_ANALYSIS_IMAGE', "false").lower()
  30. PROMPT_STATUS_CHANGE = str(os.getenv('PROMPT_STATUS_CHANGE', "Rédige un message court qui sera utilisé en tant que status sur Discord"))
  31. DELAY_TASK_UPDATE_STATUS = int(os.getenv('DELAY_TASK_UPDATE_STATUS', '30'))
  32. # Initialiser le client OpenAI asynchrone ici
  33. openai_client = AsyncOpenAI(api_key=OPENAI_API_KEY, base_url=URL_OPENAI_API)
  34. BOT_VERSION = "2.9.1"
  35. # Vérifier que les tokens et le prompt de personnalité sont récupérés
  36. if DISCORD_TOKEN is None or OPENAI_API_KEY is None or DISCORD_CHANNEL_ID is None:
  37. raise ValueError("Les tokens ou l'ID du canal ne sont pas définis dans les variables d'environnement.")
  38. if not os.path.isfile(PERSONALITY_PROMPT_FILE):
  39. raise FileNotFoundError(f"Le fichier de prompt de personnalité '{PERSONALITY_PROMPT_FILE}' est introuvable.")
  40. # Lire le prompt de personnalité depuis le fichier
  41. with open(PERSONALITY_PROMPT_FILE, 'r', encoding='utf-8') as f:
  42. PERSONALITY_PROMPT = f.read().strip()
  43. # Log configuration
  44. log_format = '%(asctime)-13s : %(name)-15s : %(levelname)-8s : %(message)s'
  45. logging.basicConfig(handlers=[logging.FileHandler("./chatbot.log", 'a', 'utf-8')], format=log_format, level=LOG_LEVEL)
  46. console = logging.StreamHandler()
  47. console.setLevel(logging.DEBUG)
  48. console.setFormatter(logging.Formatter(log_format))
  49. logger = logging.getLogger(BOT_NAME)
  50. logger.setLevel("DEBUG")
  51. logging.getLogger('').addHandler(console)
  52. httpx_logger = logging.getLogger('httpx')
  53. httpx_logger.setLevel(logging.DEBUG)
  54. urllib3.disable_warnings()
  55. # Initialiser les intents
  56. intents = discord.Intents.default()
  57. intents.message_content = True # Activer l'intent pour les contenus de message
  58. # Liste pour stocker l'historique des conversations
  59. conversation_history = []
  60. def filter_message(message):
  61. """Filtre le contenu d'un retour de modèle de language, comme pour enlever les pensées dans le cas par exemple de DeepSeek"""
  62. THOUGHT_TAGS = [
  63. "think",
  64. "analysis",
  65. "response",
  66. "reasoning"
  67. ]
  68. for tag in THOUGHT_TAGS:
  69. message = re.sub(rf"<{tag}>.*?</{tag}>", "", message, flags=re.DOTALL | re.IGNORECASE)
  70. message = re.sub(r"</s>$", "", message)
  71. return message.strip()
  72. def transform_emote(message: str, output: bool) -> str:
  73. """Remplace les smileys par les codes Discord correspondant"""
  74. list_emote = [
  75. (":hap:", "<:hap:355854929073537026>"),
  76. (":angryvault:", "<:angryvault:585550568806940672>"),
  77. (":minou:", "<:minou:358054423462936576>"),
  78. (":cetaitsur:", "<a:cetaitsur:826102032963469324>"),
  79. (":eh:", "<:eh:395979132896280576>"),
  80. (":desu:", "<:desu:388007643077410837>"),
  81. (":bave2:", "<:bave2:412252920558387221>"),
  82. (":haptriste:", "<:haptriste:358054014262181889>"),
  83. (":perplexe:", "<:perplexe:358054891274371082>"),
  84. (":sueur:", "<:sueur:358051940631838721>"),
  85. (":chien:", "<:chien:507606737646518293>"),
  86. (":kemar:", "<:kemar:419607012796792842>"),
  87. (":ouch2:", "<:ouch2:777984650710745138>"),
  88. (":coeur:", "<:coeur:355853389399195649>"),
  89. (":what:", "<:what:587019571207077928>"),
  90. # Un peu un hack mais flemme de renommer la fonction
  91. ("@Rika", "<@1284696824804016138>")
  92. ]
  93. for smiley, discord_code in list_emote:
  94. if output:
  95. message = message.replace(smiley, discord_code)
  96. else:
  97. message = message.replace(discord_code, smiley)
  98. return message
  99. def split_message(message, max_length=2000):
  100. """Divise un message en plusieurs segments de longueur maximale spécifiée."""
  101. if len(message) <= max_length:
  102. return [message]
  103. parts = []
  104. current_part = ""
  105. for line in message.split('\n'):
  106. if len(current_part) + len(line) + 1 > max_length:
  107. parts.append(current_part)
  108. current_part = line + '\n'
  109. else:
  110. current_part += line + '\n'
  111. if current_part:
  112. parts.append(current_part)
  113. return parts
  114. def load_conversation_history():
  115. global conversation_history
  116. if os.path.isfile(CONVERSATION_HISTORY_FILE):
  117. try:
  118. with open(CONVERSATION_HISTORY_FILE, 'r', encoding='utf-8') as f:
  119. loaded_history = json.load(f)
  120. # Exclure uniquement le PERSONALITY_PROMPT
  121. conversation_history = [
  122. msg for msg in loaded_history
  123. if not (msg.get("role") == "system" and msg.get("content") == PERSONALITY_PROMPT)
  124. ]
  125. logger.info(f"Historique chargé depuis {CONVERSATION_HISTORY_FILE}")
  126. except Exception as e:
  127. logger.error(f"Erreur lors du chargement de l'historique : {e}")
  128. conversation_history = []
  129. else:
  130. logger.info(f"Aucun fichier d'historique trouvé. Un nouveau fichier sera créé à {CONVERSATION_HISTORY_FILE}")
  131. def has_text(text):
  132. """Détermine si le texte fourni est non vide après suppression des espaces."""
  133. return bool(text.strip())
  134. def resize_image(image_bytes, attachment_filename=None):
  135. """Redimensionne l'image selon le mode spécifié."""
  136. with Image.open(BytesIO(image_bytes)) as img:
  137. original_format = img.format # Stocker le format original
  138. img.thumbnail((2000, 2000))
  139. buffer = BytesIO()
  140. img_format = img.format or _infer_image_format(attachment_filename)
  141. img.save(buffer, format=img_format)
  142. return buffer.getvalue()
  143. async def encode_image_from_attachment(attachment):
  144. """Encode une image depuis une pièce jointe en base64 après redimensionnement."""
  145. image_data = await attachment.read()
  146. resized_image = resize_image(image_data, attachment_filename=attachment.filename)
  147. return base64.b64encode(resized_image).decode('utf-8')
  148. def _infer_image_format(filename):
  149. """Déduit le format de l'image basé sur l'extension du fichier."""
  150. if filename:
  151. _, ext = os.path.splitext(filename)
  152. ext = ext.lower()
  153. format_mapping = {
  154. '.jpg': 'JPEG',
  155. '.jpeg': 'JPEG',
  156. '.png': 'PNG',
  157. '.gif': 'GIF',
  158. '.bmp': 'BMP',
  159. '.tiff': 'TIFF'
  160. }
  161. return format_mapping.get(ext, 'PNG')
  162. return 'PNG'
  163. # Fonction de sauvegarde de l'historique
  164. def save_conversation_history():
  165. try:
  166. with open(CONVERSATION_HISTORY_FILE, 'w', encoding='utf-8') as f:
  167. json.dump(conversation_history, f, ensure_ascii=False, indent=4)
  168. except Exception as e:
  169. logger.error(f"Erreur lors de la sauvegarde de l'historique : {e}")
  170. # Convertir l'ID du channel en entier
  171. try:
  172. discord_channel_id = int(DISCORD_CHANNEL_ID)
  173. except ValueError:
  174. raise ValueError("L'ID du channel Discord est invalide. Assurez-vous qu'il s'agit d'un entier.")
  175. class MyDiscordClient(discord.Client):
  176. async def setup_hook(self):
  177. update_status.start()
  178. async def close(self):
  179. global openai_client
  180. if openai_client is not None:
  181. await openai_client.close()
  182. openai_client = None
  183. await super().close()
  184. # Initialiser le client Discord avec les intents modifiés
  185. client_discord = MyDiscordClient(intents=intents)
  186. # Appeler la fonction pour charger l'historique au démarrage
  187. load_conversation_history()
  188. async def call_for_image_analysis(image_data, user_name, user_text=None):
  189. """Appelle l'API pour analyser une image."""
  190. user_content = (
  191. f"{user_name} a envoyé une image avec le messsage : {transform_emote(user_text, False)}."
  192. if user_text
  193. else f"{user_name} a envoyé une image."
  194. )
  195. message_history = {
  196. "role": "user",
  197. "content": [{ "type": "text", "text": user_content }]
  198. }
  199. prompt = {
  200. "role": "system",
  201. "content": PERSONALITY_PROMPT
  202. }
  203. message_to_send = {
  204. "role": "user",
  205. "content": [
  206. {
  207. "type": "text",
  208. "text": user_content
  209. },
  210. {
  211. "type": "image_url",
  212. "image_url": {
  213. "url": f"data:image/jpeg;base64,{image_data}"
  214. }
  215. }
  216. ]
  217. }
  218. if HISTORY_ANALYSIS_IMAGE != "false":
  219. messages = [prompt] + conversation_history + [message_to_send]
  220. else:
  221. messages = [prompt] + [message_to_send]
  222. await add_to_conversation_history(message_history)
  223. analysis = await openai_client.chat.completions.create(
  224. model=MODEL,
  225. stream=False,
  226. messages=messages,
  227. temperature=TEMPERATURE
  228. )
  229. if not analysis or not getattr(analysis, "choices", None):
  230. raise RuntimeError("Réponse API invalide ou vide")
  231. else:
  232. logger.info(f"Analyse de l'image par l'API : {analysis.choices[0].message.content}")
  233. await add_to_conversation_history({
  234. "role": "assistant",
  235. "content": analysis.choices[0].message.content
  236. })
  237. return analysis.choices[0].message.content
  238. async def read_text_file(attachment):
  239. """Lit le contenu d'un fichier texte attaché."""
  240. file_bytes = await attachment.read()
  241. result = from_bytes(file_bytes).best()
  242. if result is None:
  243. raise ValueError("Impossible de détecter l'encodage du fichier")
  244. return str(result)
  245. async def call_openai_api(user_text, user_name):
  246. # Préparer le contenu pour l'appel API
  247. message_to_send = {
  248. "role": "user",
  249. "content": [
  250. {
  251. "type": "text",
  252. "text": f"{user_name} dit : {transform_emote(user_text, False)}"
  253. }
  254. ]
  255. }
  256. # Assembler les messages avec le prompt de personnalité en premier
  257. messages = [
  258. {"role": "system", "content": PERSONALITY_PROMPT}
  259. ] + conversation_history + [message_to_send]
  260. try:
  261. response = await openai_client.chat.completions.create(
  262. model=MODEL,
  263. stream=False,
  264. messages=messages,
  265. temperature=TEMPERATURE
  266. )
  267. await add_to_conversation_history(message_to_send)
  268. # Ajouter la réponse de l'IA directement à l'historique
  269. await add_to_conversation_history({
  270. "role": "assistant",
  271. "content": response.choices[0].message.content
  272. })
  273. return response
  274. except Exception as e:
  275. logger.error(f"Erreur durant l'appel de l'API : {e}")
  276. return None
  277. async def call_openai_api_system(system_text):
  278. # Préparer le contenu pour l'appel API
  279. message = {
  280. "role": "system",
  281. "content": PERSONALITY_PROMPT + f"\n\nPour cette réponse uniquement : {system_text}"
  282. }
  283. response = await openai_client.chat.completions.create(
  284. model=MODEL,
  285. stream=False,
  286. messages=[message],
  287. temperature=1.2
  288. )
  289. return response
  290. @client_discord.event
  291. async def on_ready():
  292. logger.info(f'{BOT_NAME} connecté en tant que {client_discord.user}')
  293. logger.info(f'Utilisation du modèle {MODEL}')
  294. if not conversation_history:
  295. logger.info("Aucun historique trouvé. L'historique commence vide.")
  296. # Envoyer un message de version dans le canal Discord
  297. channel = client_discord.get_channel(discord_channel_id)
  298. if channel:
  299. if BOOT_MESSAGE != "false":
  300. try:
  301. embed = discord.Embed(
  302. title=f"Bot Démarré",
  303. description=f"🎉 {BOT_NAME} est en ligne ! Version {BOT_VERSION}\nUtilisation du modèle: **{MODEL}**",
  304. color=0x2222aa # Bleu
  305. )
  306. await channel.send(embed=embed)
  307. logger.info(f"Message de connexion envoyé dans le canal ID {discord_channel_id}")
  308. except discord.Forbidden:
  309. logger.error(f"Permissions insuffisantes pour envoyer des messages dans le canal ID {discord_channel_id}.")
  310. except discord.HTTPException as e:
  311. logger.error(f"Erreur lors de l'envoi du message de connexion : {e}")
  312. else:
  313. logger.error(f"Canal avec ID {discord_channel_id} non trouvé.")
  314. @client_discord.event
  315. async def on_message(message):
  316. global conversation_history
  317. # Vérifier si le message provient du canal autorisé
  318. if message.channel.id != discord_channel_id:
  319. return
  320. # Ignorer les messages du bot lui-même
  321. if message.author == client_discord.user:
  322. return
  323. user_text = message.content.strip()
  324. file_content = None
  325. attachment_filename = None
  326. # Vérifier si le message est la commande de réinitialisation
  327. if user_text.lower() == "!reset_history":
  328. # Vérifier si l'utilisateur a les permissions administratives
  329. if not message.author.guild_permissions.administrator:
  330. await message.channel.send("❌ Vous n'avez pas la permission d'utiliser cette commande.")
  331. return
  332. conversation_history = []
  333. save_conversation_history()
  334. await message.channel.send("✅ L'historique des conversations a été réinitialisé.")
  335. logger.info(f"Historique des conversations réinitialisé par {message.author}.")
  336. return # Arrêter le traitement du message après la réinitialisation
  337. # Variable pour stocker si le message contient un fichier
  338. has_file = False
  339. image_data = None
  340. # Vérifier s'il y a une pièce jointe
  341. if message.attachments:
  342. for attachment in message.attachments:
  343. # Vérifier si c'est un fichier avec une extension autorisée
  344. if attachment.content_type and attachment.content_type.startswith('image/'):
  345. try:
  346. image_data = await encode_image_from_attachment(attachment)
  347. break
  348. except Exception as e:
  349. await message.channel.send("Il semble qu'il y ai un souci avec ton image, je ne peux pas l'ouvrir.")
  350. logger.error(f"Erreur lors de la conversion de l'image : {e}")
  351. else:
  352. try:
  353. file_content = await read_text_file(attachment)
  354. attachment_filename = attachment.filename
  355. break
  356. except Exception as e:
  357. await message.channel.send("Désolé, je n'ai pas pu lire ton fichier.")
  358. logger.error(f"Erreur lors de la lecture d'une pièce jointe : {e}")
  359. # Traitement des images
  360. if image_data:
  361. logger.debug(image_data)
  362. has_user_text = has_text(user_text)
  363. user_text_to_use = user_text if has_user_text else None
  364. temp_msg = await message.channel.send(f"*{BOT_NAME} observe l'image...*")
  365. try:
  366. # Analyser l'image avec l'API
  367. analysis = await call_for_image_analysis(image_data, message.author.name, user_text=user_text_to_use)
  368. reply = filter_message(analysis)
  369. reply = transform_emote(analysis, True)
  370. if analysis:
  371. await temp_msg.delete()
  372. await message.channel.send(reply)
  373. if has_user_text:
  374. user_message_text = f"{message.author.name} a envoyé une image avec le messsage : \"{user_text}\"."
  375. else:
  376. user_message_text = f"{message.author.name} a envoyé une image."
  377. user_message = {
  378. "role": "user",
  379. "content": f"{user_message_text}"
  380. }
  381. assistant_message = {
  382. "role": "assistant",
  383. "content": analysis
  384. }
  385. await add_to_conversation_history(user_message)
  386. await add_to_conversation_history(assistant_message)
  387. else:
  388. await temp_msg.delete()
  389. await message.channel.send("Désolé, je n'ai pas pu analyser l'image.")
  390. except Exception as e:
  391. await temp_msg.delete()
  392. await message.channel.send("Une erreur est survenue lors du traitement de l'image.")
  393. logger.error(f"Erreur lors du traitement de l'image: {e}")
  394. return # Ne pas continuer le traitement après une image
  395. # Ajouter le contenu du fichier à la requête si présent
  396. if file_content:
  397. user_text += f"\nPièce jointe ajoutée au message, contenu du fichier '{attachment.filename}':\n{file_content}"
  398. # Vérifier si le texte n'est pas vide après ajout du contenu du fichier
  399. if not has_text(user_text):
  400. return # Ne pas appeler l'API si le texte est vide
  401. async with message.channel.typing():
  402. try:
  403. # Appeler l'API
  404. result = await call_openai_api(user_text, message.author.name)
  405. if result:
  406. reply = result.choices[0].message.content
  407. reply = filter_message(reply)
  408. reply = transform_emote(reply, True)
  409. message_parts = split_message(reply)
  410. for part in message_parts:
  411. await message.channel.send(part)
  412. # Afficher dans la console
  413. logging.info(f"Réponse envoyée. ({len(message_parts)} message(s))")
  414. except Exception as e:
  415. await message.channel.send("Franchement, je sais pas quoi te répondre. <:haptriste:358054014262181889>")
  416. logger.error(f"Erreur lors du traitement du texte: {e}")
  417. async def add_to_conversation_history(new_message):
  418. global conversation_history
  419. conversation_history.append(new_message)
  420. save_conversation_history()
  421. logger.debug(f"Message ajouté à l'historique. Taille actuelle : {len(conversation_history)}")
  422. if len(conversation_history) > CONVERSATION_HISTORY_SIZE:
  423. logger.info(f"Limite de {CONVERSATION_HISTORY_SIZE} messages atteinte.")
  424. excess_messages = len(conversation_history) - CONVERSATION_HISTORY_SIZE
  425. # Supprimer les messages les plus anciens
  426. del conversation_history[:excess_messages]
  427. save_conversation_history()
  428. logger.info(f"{excess_messages} messages les plus anciens ont été supprimés.")
  429. @tasks.loop(minutes=DELAY_TASK_UPDATE_STATUS)
  430. async def update_status():
  431. try:
  432. reply = await call_openai_api_system(PROMPT_STATUS_CHANGE)
  433. status = reply.choices[0].message.content
  434. logger.info(f"Nouveau status Discord: '{status}'")
  435. await client_discord.change_presence(activity=discord.CustomActivity(name=status))
  436. except Exception as e:
  437. logger.warning(f"Impossible de changer le status Discord : {e}")
  438. @update_status.before_loop
  439. async def before_update_status():
  440. await client_discord.wait_until_ready()
  441. logger.info("Démarrage de la tâche update_status")
  442. # Démarrer le bot Discord
  443. client_discord.run(DISCORD_TOKEN)