2023-03-18 02:14:45 -06:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
import argparse
|
|
|
|
import asyncio
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import time
|
|
|
|
import traceback
|
|
|
|
from pathlib import Path
|
2023-03-18 13:05:00 -06:00
|
|
|
from uuid import uuid4
|
2023-03-18 02:14:45 -06:00
|
|
|
|
|
|
|
import openai
|
|
|
|
import yaml
|
|
|
|
from aiohttp import ClientConnectionError, ServerDisconnectedError
|
|
|
|
from nio import InviteMemberEvent, JoinResponse, LocalProtocolError, MegolmEvent, RoomMessageText
|
|
|
|
|
|
|
|
from matrix_gpt import MatrixNioGPTHelper
|
|
|
|
from matrix_gpt.bot.callbacks import Callbacks
|
|
|
|
from matrix_gpt.bot.storage import Storage
|
|
|
|
from matrix_gpt.config import check_config_value_exists
|
|
|
|
|
|
|
|
script_directory = os.path.abspath(os.path.dirname(__file__))
|
|
|
|
|
|
|
|
logging.basicConfig()
|
|
|
|
logger = logging.getLogger('MatrixGPT')
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(description='MatrixGPT Bot')
|
|
|
|
parser.add_argument('--config', default=Path(script_directory, 'config.yaml'), help='Path to config.yaml if it is not located next to this executable.')
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
# Load config
|
|
|
|
if not Path(args.config).exists():
|
|
|
|
print('Config file does not exist:', args.config)
|
|
|
|
sys.exit(1)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
with open(args.config, 'r') as file:
|
|
|
|
config_data = yaml.safe_load(file)
|
|
|
|
except Exception as e:
|
|
|
|
print(f'Failed to load config file: {e}')
|
|
|
|
sys.exit(1)
|
|
|
|
|
2023-03-19 14:46:42 -06:00
|
|
|
# Lazy way to validate config
|
2023-03-18 02:14:45 -06:00
|
|
|
check_config_value_exists(config_data, 'bot_auth', dict)
|
|
|
|
check_config_value_exists(config_data['bot_auth'], 'username')
|
|
|
|
check_config_value_exists(config_data['bot_auth'], 'password')
|
|
|
|
check_config_value_exists(config_data['bot_auth'], 'homeserver')
|
|
|
|
check_config_value_exists(config_data['bot_auth'], 'store_path')
|
|
|
|
check_config_value_exists(config_data, 'allowed_to_chat')
|
|
|
|
check_config_value_exists(config_data, 'allowed_to_invite', allow_empty=True)
|
|
|
|
check_config_value_exists(config_data, 'data_storage')
|
2023-03-31 23:08:27 -06:00
|
|
|
check_config_value_exists(config_data, 'command')
|
2023-03-18 02:14:45 -06:00
|
|
|
|
2023-03-19 15:22:05 -06:00
|
|
|
check_config_value_exists(config_data, 'logging')
|
|
|
|
check_config_value_exists(config_data['logging'], 'log_level')
|
|
|
|
|
2023-03-19 14:46:42 -06:00
|
|
|
check_config_value_exists(config_data, 'openai')
|
|
|
|
check_config_value_exists(config_data['openai'], 'api_key')
|
|
|
|
check_config_value_exists(config_data['openai'], 'model')
|
|
|
|
|
2023-03-31 23:08:27 -06:00
|
|
|
gpt4_enabled = True if config_data['command'].get('gpt4_prefix') else False
|
|
|
|
logger.info(f'GPT4 enabled? {gpt4_enabled}')
|
|
|
|
|
|
|
|
command_prefixes = {}
|
|
|
|
for k, v in config_data['command'].items():
|
|
|
|
command_prefixes[k] = v
|
|
|
|
|
2023-03-18 15:54:00 -06:00
|
|
|
|
2023-03-18 02:14:45 -06:00
|
|
|
# check_config_value_exists(config_data, 'autojoin_rooms')
|
|
|
|
|
|
|
|
def retry(msg=None):
|
|
|
|
if msg:
|
|
|
|
logger.warning(f'{msg}, retrying in 15s...')
|
|
|
|
else:
|
|
|
|
logger.warning(f'Retrying in 15s...')
|
|
|
|
time.sleep(15)
|
|
|
|
|
|
|
|
|
|
|
|
async def main():
|
2023-03-19 15:22:05 -06:00
|
|
|
if config_data['logging']['log_level'] == 'info':
|
|
|
|
log_level = logging.INFO
|
|
|
|
elif config_data['logging']['log_level'] == 'debug':
|
|
|
|
log_level = logging.DEBUG
|
|
|
|
elif config_data['logging']['log_level'] == 'warning':
|
|
|
|
log_level = logging.WARNING
|
|
|
|
elif config_data['logging']['log_level'] == 'critical':
|
|
|
|
log_level = logging.CRITICAL
|
|
|
|
else:
|
|
|
|
log_level = logging.INFO
|
|
|
|
logger.setLevel(log_level)
|
|
|
|
|
2023-03-18 13:05:00 -06:00
|
|
|
# Logging in with a new device each time seems to fix encryption errors
|
2023-03-18 13:54:13 -06:00
|
|
|
device_id = config_data['bot_auth'].get('device_id', str(uuid4()))
|
2023-03-18 13:05:00 -06:00
|
|
|
|
2023-03-19 15:24:02 -06:00
|
|
|
matrix_helper = MatrixNioGPTHelper(
|
|
|
|
auth_file=Path(config_data['bot_auth']['store_path'], 'bot_auth.json'),
|
|
|
|
user_id=config_data['bot_auth']['username'],
|
|
|
|
passwd=config_data['bot_auth']['password'],
|
|
|
|
homeserver=config_data['bot_auth']['homeserver'],
|
|
|
|
store_path=config_data['bot_auth']['store_path'],
|
|
|
|
device_id=device_id,
|
|
|
|
)
|
2023-03-18 02:14:45 -06:00
|
|
|
client = matrix_helper.client
|
|
|
|
|
2023-03-19 14:46:42 -06:00
|
|
|
openai.api_key = config_data['openai']['api_key']
|
2023-03-18 02:14:45 -06:00
|
|
|
|
|
|
|
storage = Storage(Path(config_data['data_storage'], 'matrixgpt.db'))
|
|
|
|
|
|
|
|
# Set up event callbacks
|
2023-03-19 15:24:02 -06:00
|
|
|
callbacks = Callbacks(client, storage,
|
|
|
|
openai_obj=openai,
|
2023-03-31 23:08:27 -06:00
|
|
|
command_prefixes=command_prefixes,
|
2023-03-19 15:24:02 -06:00
|
|
|
openai_model=config_data['openai']['model'],
|
|
|
|
reply_in_thread=config_data.get('reply_in_thread', False),
|
|
|
|
allowed_to_invite=config_data['allowed_to_invite'],
|
|
|
|
allowed_to_chat=config_data['allowed_to_chat'],
|
|
|
|
log_full_response=config_data['logging'].get('log_full_response', False),
|
|
|
|
system_prompt=config_data['openai'].get('system_prompt'),
|
|
|
|
injected_system_prompt=config_data['openai'].get('injected_system_prompt', False),
|
2023-03-22 16:44:17 -06:00
|
|
|
openai_temperature=config_data['openai'].get('temperature', 0),
|
2023-03-31 23:08:27 -06:00
|
|
|
gpt4_enabled=gpt4_enabled,
|
2023-03-22 16:44:17 -06:00
|
|
|
log_level=log_level
|
2023-03-19 15:24:02 -06:00
|
|
|
)
|
2023-03-18 02:14:45 -06:00
|
|
|
client.add_event_callback(callbacks.message, RoomMessageText)
|
|
|
|
client.add_event_callback(callbacks.invite_event_filtered_callback, InviteMemberEvent)
|
|
|
|
client.add_event_callback(callbacks.decryption_failure, MegolmEvent)
|
|
|
|
# client.add_event_callback(callbacks.unknown, UnknownEvent)
|
|
|
|
|
|
|
|
# Keep trying to reconnect on failure (with some time in-between)
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
# Try to login with the configured username/password
|
|
|
|
try:
|
|
|
|
login_response = await matrix_helper.login()
|
|
|
|
|
|
|
|
# Check if login failed
|
|
|
|
if not login_response[0]:
|
|
|
|
logger.error(f'Failed to login: {login_response[1].message}\n{vars(login_response[1])}')
|
|
|
|
retry()
|
|
|
|
return False
|
|
|
|
except LocalProtocolError as e:
|
|
|
|
# There's an edge case here where the user hasn't installed the correct C
|
|
|
|
# dependencies. In that case, a LocalProtocolError is raised on login.
|
|
|
|
logger.fatal(f'Failed to login:\n{e}')
|
|
|
|
retry()
|
|
|
|
return False
|
|
|
|
|
|
|
|
# Login succeeded!
|
2023-03-18 13:54:13 -06:00
|
|
|
logger.info(f"Logged in as {client.user_id} using device {device_id}.")
|
2023-03-18 02:14:45 -06:00
|
|
|
if config_data.get('autojoin_rooms'):
|
|
|
|
for room in config_data.get('autojoin_rooms'):
|
|
|
|
r = await client.join(room)
|
|
|
|
if not isinstance(r, JoinResponse):
|
|
|
|
logger.critical(f'Failed to join room {room}: {vars(r)}')
|
2023-03-18 13:05:00 -06:00
|
|
|
time.sleep(1.5)
|
|
|
|
|
|
|
|
# Log out old devices to keep the session clean
|
2023-03-18 14:37:50 -06:00
|
|
|
if config_data.get('logout_other_devices', False):
|
|
|
|
logger.info('Logging out other devices...')
|
|
|
|
devices = list((await client.devices()).devices)
|
|
|
|
device_list = [x.id for x in devices]
|
|
|
|
if device_id in device_list:
|
|
|
|
device_list.remove(device_id)
|
2023-03-19 15:22:05 -06:00
|
|
|
x = await client.delete_devices(device_list, {"type": "m.login.password", "user": config_data['bot_auth']['username'], "password": config_data['bot_auth']['password']})
|
2023-03-18 14:37:50 -06:00
|
|
|
logger.info(f'Logged out: {device_list}')
|
2023-03-18 02:14:45 -06:00
|
|
|
|
|
|
|
await client.sync_forever(timeout=10000, full_state=True)
|
|
|
|
except (ClientConnectionError, ServerDisconnectedError):
|
|
|
|
logger.warning("Unable to connect to homeserver, retrying in 15s...")
|
|
|
|
time.sleep(15)
|
2023-03-18 13:32:04 -06:00
|
|
|
except KeyboardInterrupt:
|
|
|
|
await client.close()
|
|
|
|
sys.exit()
|
2023-03-18 13:05:00 -06:00
|
|
|
except Exception:
|
|
|
|
logger.critical(traceback.format_exc())
|
|
|
|
logger.critical('Sleeping 5s...')
|
|
|
|
time.sleep(5)
|
2023-03-18 02:14:45 -06:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
asyncio.run(main())
|
2023-03-18 15:54:00 -06:00
|
|
|
except KeyboardInterrupt:
|
|
|
|
sys.exit()
|
2023-03-18 02:14:45 -06:00
|
|
|
except Exception:
|
|
|
|
logger.critical(traceback.format_exc())
|
2023-03-18 03:21:03 -06:00
|
|
|
time.sleep(5)
|