upgrading script to v2 with history
This commit is contained in:
		| @@ -1,111 +1,100 @@ | ||||
| import os | ||||
| import discord | ||||
| from discord.ext import commands | ||||
| import openai | ||||
| from openai import OpenAI | ||||
| import os | ||||
|  | ||||
| from collections import deque | ||||
| from dotenv import load_dotenv | ||||
|  | ||||
| # Load environment variables | ||||
| load_dotenv() | ||||
|  | ||||
| # Get environment variables | ||||
| DISCORD_TOKEN = os.getenv('DISCORD_TOKEN') | ||||
| OPENAI_API_KEY = os.getenv('OPENAI_API_KEY') | ||||
| OPENWEBUI_API_BASE = os.getenv('OPENWEBUI_API_BASE') | ||||
| MODEL_NAME = os.getenv('MODEL_NAME') | ||||
|  | ||||
| # Configure OpenAI client to point to OpenWebUI | ||||
| client = OpenAI( | ||||
|     api_key=os.getenv('OPENAI_API_KEY'), | ||||
|     base_url=os.getenv('OPENWEBUI_API_BASE')  # e.g., "http://localhost:8080/v1" | ||||
| ) | ||||
|  | ||||
| # Configure OpenAI | ||||
| # TODO: The 'openai.api_base' option isn't read in the client API. You will need to pass it when you instantiate the client, e.g. 'OpenAI(base_url=OPENWEBUI_API_BASE)' | ||||
| # openai.api_base = OPENWEBUI_API_BASE | ||||
|  | ||||
| # Initialize Discord bot | ||||
| intents = discord.Intents.default() | ||||
| intents.message_content = True | ||||
| bot = commands.Bot(command_prefix="!", intents=intents) | ||||
| intents.messages = True | ||||
| bot = commands.Bot(command_prefix='!', intents=intents) | ||||
|  | ||||
| # Add a dictionary to store conversation histories | ||||
| conversation_histories = {} | ||||
| DEFAULT_HISTORY_LIMIT = 50 | ||||
| MAX_HISTORY_LIMIT = 200 | ||||
| # Message history cache | ||||
| channel_history = {} | ||||
|  | ||||
| async def get_ai_response(prompt, channel_id=None): | ||||
|     try: | ||||
| async def get_chat_history(channel, limit=100): | ||||
|     messages = [] | ||||
|     async for message in channel.history(limit=limit): | ||||
|         messages.append(f"{message.author.name}: {message.content}") | ||||
|     return "\n".join(reversed(messages)) | ||||
|  | ||||
|         # Always include history if it exists for this channel | ||||
|         if channel_id in conversation_histories: | ||||
|             history = conversation_histories[channel_id][-400:]  # Get last 400 messages (200 exchanges) | ||||
| async def get_ai_response(context, user_message): | ||||
|     formatted_prompt = f"##CONTEXT##\n{context}\n##ENDCONTEXT##\n\n{user_message}" | ||||
|  | ||||
|             # Format the history as a context block | ||||
|             history_text = "\n".join([ | ||||
|                 f"{'User: ' if msg['role'] == 'user' else 'Assistant: '}{msg['content']}" | ||||
|                 for msg in history | ||||
|     try: | ||||
|         response = client.chat.completions.create(model=MODEL_NAME, | ||||
|         messages=[ | ||||
|             {"role": "user", "content": formatted_prompt} | ||||
|         ]) | ||||
|              | ||||
|             formatted_prompt = f"{prompt}\n##CONTEXT##\n{history_text}\n##ENDCONTEXT##" | ||||
|         else: | ||||
|             formatted_prompt = prompt | ||||
|  | ||||
|         response = client.chat.completions.create( | ||||
|             model=os.getenv('MODEL_NAME') or "us.anthropic.claude-3-5-sonnet-20241022-v2:0", | ||||
|             messages=[{"role": "user", "content": formatted_prompt}], | ||||
|             temperature=0.7, | ||||
|             max_tokens=500 | ||||
|         ) | ||||
|          | ||||
|         # Store the conversation history if channel_id is provided | ||||
|         if channel_id: | ||||
|             if channel_id not in conversation_histories: | ||||
|                 conversation_histories[channel_id] = [] | ||||
|             conversation_histories[channel_id].extend([ | ||||
|                 {"role": "user", "content": prompt}, | ||||
|                 {"role": "assistant", "content": response.choices[0].message.content} | ||||
|             ]) | ||||
|              | ||||
|             # Always keep last 400 messages (200 exchanges) | ||||
|             if len(conversation_histories[channel_id]) > 400: | ||||
|                 conversation_histories[channel_id] = conversation_histories[channel_id][-400:] | ||||
|          | ||||
|         return response.choices[0].message.content | ||||
|     except Exception as e: | ||||
|         print(f"Error getting AI response: {e}") | ||||
|         return "Sorry, I encountered an error while processing your request." | ||||
|         return f"Error: {str(e)}" | ||||
|  | ||||
| @bot.event | ||||
| async def on_ready(): | ||||
|     print(f'{bot.user} has connected to Discord!') | ||||
|  | ||||
| @bot.event | ||||
| async def on_message(message): | ||||
|     # Ignore messages from the bot itself | ||||
|     if message.author == bot.user: | ||||
|         return | ||||
|  | ||||
|     # Respond to DMs or when mentioned in a server | ||||
|     if isinstance(message.channel, discord.DMChannel) or bot.user in message.mentions: | ||||
|         # For mentions, remove the bot mention from the message | ||||
|     should_respond = False | ||||
|  | ||||
|     # Check if bot was mentioned | ||||
|     if bot.user in message.mentions: | ||||
|             prompt = message.content.replace(f'<@{bot.user.id}>', '').strip() | ||||
|         else: | ||||
|             prompt = message.content.strip() | ||||
|         should_respond = True | ||||
|  | ||||
|         if not prompt: | ||||
|             await message.channel.send("Hello! How can I help you?") | ||||
|             return | ||||
|     # Check if message is a DM | ||||
|     if isinstance(message.channel, discord.DMChannel): | ||||
|         should_respond = True | ||||
|  | ||||
|     if should_respond: | ||||
|         async with message.channel.typing(): | ||||
|             response = await get_ai_response( | ||||
|                 prompt, | ||||
|                 channel_id=str(message.channel.id) | ||||
|             ) | ||||
|             # Get chat history | ||||
|             history = await get_chat_history(message.channel) | ||||
|  | ||||
|             if len(response) > 2000: | ||||
|                 chunks = [response[i:i+2000] for i in range(0, len(response), 2000)] | ||||
|                 for chunk in chunks: | ||||
|                     await message.channel.send(chunk) | ||||
|             else: | ||||
|                 await message.channel.send(response) | ||||
|             # Remove bot mention from the message | ||||
|             user_message = message.content.replace(f'<@{bot.user.id}>', '').strip() | ||||
|  | ||||
|             # Get AI response | ||||
|             response = await get_ai_response(history, user_message) | ||||
|  | ||||
|             # Send response | ||||
|             await message.reply(response) | ||||
|  | ||||
|     await bot.process_commands(message) | ||||
|  | ||||
| def main(): | ||||
|     # Get the Discord token from environment variables | ||||
|     discord_token = os.getenv('DISCORD_TOKEN') | ||||
|     if not discord_token: | ||||
|         raise ValueError("Discord token not found in environment variables") | ||||
|     if not all([DISCORD_TOKEN, OPENAI_API_KEY, OPENWEBUI_API_BASE, MODEL_NAME]): | ||||
|         print("Error: Missing required environment variables") | ||||
|         return | ||||
|  | ||||
|     # Run the bot | ||||
|     bot.run(discord_token) | ||||
|     bot.run(DISCORD_TOKEN) | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										18
									
								
								v2/bot.py
									
									
									
									
									
								
							
							
						
						
									
										18
									
								
								v2/bot.py
									
									
									
									
									
								
							| @@ -2,6 +2,8 @@ import os | ||||
| import discord | ||||
| from discord.ext import commands | ||||
| import openai | ||||
| from openai import OpenAI | ||||
|  | ||||
| from collections import deque | ||||
| from dotenv import load_dotenv | ||||
|  | ||||
| @@ -14,9 +16,15 @@ OPENAI_API_KEY = os.getenv('OPENAI_API_KEY') | ||||
| OPENWEBUI_API_BASE = os.getenv('OPENWEBUI_API_BASE') | ||||
| MODEL_NAME = os.getenv('MODEL_NAME') | ||||
|  | ||||
| # Configure OpenAI client to point to OpenWebUI | ||||
| client = OpenAI( | ||||
|     api_key=os.getenv('OPENAI_API_KEY'), | ||||
|     base_url=os.getenv('OPENWEBUI_API_BASE')  # e.g., "http://localhost:8080/v1" | ||||
| ) | ||||
|  | ||||
| # Configure OpenAI | ||||
| openai.api_key = OPENAI_API_KEY | ||||
| openai.api_base = OPENWEBUI_API_BASE | ||||
| # TODO: The 'openai.api_base' option isn't read in the client API. You will need to pass it when you instantiate the client, e.g. 'OpenAI(base_url=OPENWEBUI_API_BASE)' | ||||
| # openai.api_base = OPENWEBUI_API_BASE | ||||
|  | ||||
| # Initialize Discord bot | ||||
| intents = discord.Intents.default() | ||||
| @@ -37,12 +45,10 @@ async def get_ai_response(context, user_message): | ||||
|     formatted_prompt = f"##CONTEXT##\n{context}\n##ENDCONTEXT##\n\n{user_message}" | ||||
|  | ||||
|     try: | ||||
|         response = openai.ChatCompletion.create( | ||||
|             model=MODEL_NAME, | ||||
|         response = client.chat.completions.create(model=MODEL_NAME, | ||||
|         messages=[ | ||||
|             {"role": "user", "content": formatted_prompt} | ||||
|             ] | ||||
|         ) | ||||
|         ]) | ||||
|         return response.choices[0].message.content | ||||
|     except Exception as e: | ||||
|         return f"Error: {str(e)}" | ||||
|   | ||||
		Reference in New Issue
	
	Block a user