matrix_chatgpt_bot/bot.py

157 lines
5.8 KiB
Python
Raw Normal View History

2023-03-05 14:07:25 +00:00
import sys
import asyncio
2023-03-05 14:07:25 +00:00
import re
import os
from typing import Optional
from nio import AsyncClient, MatrixRoom, RoomMessageText, LoginResponse, AsyncClientConfig
from nio.store.database import SqliteStore
from ask_gpt import ask
from send_message import send_room_message
from v3 import Chatbot
2023-03-10 13:43:18 +00:00
from log import getlogger
2023-03-10 15:45:38 +00:00
from bing import BingBot
"""
free api_endpoint from https://github.com/ayaka14732/ChatGPTAPIFree
"""
2023-03-10 15:45:38 +00:00
chatgpt_api_endpoint_list = {
"free": "https://chatgpt-api.shn.hk/v1/",
"paid": "https://api.openai.com/v1/chat/completions"
}
2023-03-10 13:43:18 +00:00
logger = getlogger()
2023-03-05 14:07:25 +00:00
class Bot:
def __init__(
2023-03-10 15:45:38 +00:00
self,
homeserver: str,
user_id: str,
password: str,
device_id: str,
api_key: Optional[str] = "",
room_id: Optional[str] = '',
bing_api_endpoint: Optional[str] = '',
2023-03-12 15:24:05 +00:00
access_token: Optional[str] = '',
2023-03-05 14:07:25 +00:00
):
self.homeserver = homeserver
self.user_id = user_id
self.password = password
self.device_id = device_id
self.room_id = room_id
self.api_key = api_key
2023-03-10 15:45:38 +00:00
self.bing_api_endpoint = bing_api_endpoint
2023-03-05 14:07:25 +00:00
# initialize AsyncClient object
self.store_path = os.getcwd()
self.config = AsyncClientConfig(store=SqliteStore,
store_name="bot",
store_sync_tokens=True,
)
self.client = AsyncClient(self.homeserver, user=self.user_id, device_id=self.device_id,
config=self.config, store_path=self.store_path)
2023-03-12 15:24:05 +00:00
if access_token != '':
self.client.access_token = access_token
2023-03-05 14:07:25 +00:00
# regular expression to match keyword [!gpt {prompt}] [!chat {prompt}]
self.gpt_prog = re.compile(r"^\s*!gpt\s*(.+)$")
self.chat_prog = re.compile(r"^\s*!chat\s*(.+)$")
2023-03-10 15:45:38 +00:00
self.bing_prog = re.compile(r"^\s*!bing\s*(.+)$")
# initialize chatbot and chatgpt_api_endpoint
if self.api_key != '':
self.chatbot = Chatbot(api_key=self.api_key)
2023-03-05 14:07:25 +00:00
2023-03-10 15:45:38 +00:00
self.chatgpt_api_endpoint = chatgpt_api_endpoint_list['paid']
# request header for !gpt command
self.headers = {
"Content-Type": "application/json",
"Authorization": "Bearer " + self.api_key,
}
else:
2023-03-10 15:45:38 +00:00
self.chatgpt_api_endpoint = chatgpt_api_endpoint_list['free']
self.headers = {
"Content-Type": "application/json",
}
2023-03-10 15:45:38 +00:00
# initialize bingbot
if self.bing_api_endpoint != '':
self.bingbot = BingBot(bing_api_endpoint)
2023-03-05 14:07:25 +00:00
# message_callback event
async def message_callback(self, room: MatrixRoom, event: RoomMessageText) -> None:
2023-03-10 13:43:18 +00:00
# print info to console
print(
f"Message received in room {room.display_name}\n"
f"{room.user_name(event.sender)} | {event.body}"
)
# remove newline character from event.body
event.body = re.sub("\r\n|\r|\n", " ", event.body)
if self.room_id == '':
room_id = room.room_id
else:
room_id = self.room_id
2023-03-10 15:45:38 +00:00
2023-03-05 14:07:25 +00:00
# chatgpt
n = self.chat_prog.match(event.body)
if n:
if self.api_key != '':
# sending typing state
await self.client.room_typing(room_id)
prompt = n.group(1)
try:
# run synchronous function in different thread
text = await asyncio.to_thread(self.chatbot.ask, prompt)
text = text.strip()
await send_room_message(self.client, room_id, send_text=text)
except Exception as e:
2023-03-10 13:43:18 +00:00
logger.error("Error", exc_info=True)
print(f"Error: {e}")
pass
else:
2023-03-10 13:43:18 +00:00
logger.warning("No API_KEY provided")
await send_room_message(self.client, room_id, send_text="API_KEY not provided")
2023-03-05 14:07:25 +00:00
2023-03-10 11:19:49 +00:00
m = self.gpt_prog.match(event.body)
if m:
# sending typing state
await self.client.room_typing(room_id)
prompt = m.group(1)
try:
2023-03-10 15:45:38 +00:00
# timeout 30s
text = await asyncio.wait_for(ask(prompt, self.chatgpt_api_endpoint, self.headers), timeout=30)
2023-03-10 11:19:49 +00:00
except TimeoutError:
2023-03-10 13:43:18 +00:00
logger.error("timeoutException", exc_info=True)
text = "Timeout error"
2023-03-10 11:19:49 +00:00
text = text.strip()
await send_room_message(self.client, room_id, send_text=text)
2023-03-10 15:45:38 +00:00
# bing ai
if self.bing_api_endpoint != '':
b = self.bing_prog.match(event.body)
if b:
# sending typing state
await self.client.room_typing(room_id)
prompt = b.group(1)
try:
# timeout 30s
2023-03-14 01:49:21 +00:00
text = await asyncio.wait_for(self.bingbot.ask_bing(prompt), timeout=120)
2023-03-10 15:45:38 +00:00
except TimeoutError:
logger.error("timeoutException", exc_info=True)
text = "Timeout error"
text = text.strip()
await send_room_message(self.client, room_id, send_text=text)
2023-03-05 14:07:25 +00:00
# bot login
async def login(self) -> None:
2023-03-12 15:24:05 +00:00
try:
resp = await self.client.login(password=self.password)
if not isinstance(resp, LoginResponse):
logger.error("Login Failed")
print(f"Login Failed: {resp}")
sys.exit(1)
except Exception as e:
logger.error("Error Exception", exc_info=True)
2023-03-05 14:07:25 +00:00
# sync messages in the room
async def sync_forever(self, timeout=30000):
self.client.add_event_callback(self.message_callback, RoomMessageText)
await self.client.sync_forever(timeout=timeout, full_state=True)