Chatbot
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

139 lines
4.3 KiB

2 years ago
#!/usr/bin/env python3
import asyncio
import os, sys
import json
from .utilities.config_parser import read_config
from .bot.core import ChatBot
from .connections.matrix import ChatClient
from .connections.webui import WebUI
2 years ago
import traceback
import signal
import functools
2 years ago
import logging
logger = logging.getLogger(__name__)
DATA_DIR = './.data'
bots = []
2 years ago
async def main() -> None:
config = read_config('bot.conf')
if config.has_option('DEFAULT', 'log_level'):
log_level = config['DEFAULT']['log_level']
if log_level == 'DEBUG':
logging.basicConfig(level=logging.DEBUG)
elif log_level == 'INFO':
logging.basicConfig(level=logging.INFO)
elif log_level == 'WARNING':
logging.basicConfig(level=logging.WARNING)
elif log_level == 'ERROR':
logging.basicConfig(level=logging.ERROR)
elif log_level == 'CRITICAL':
logging.basicConfig(level=logging.CRITICAL)
# loop = asyncio.get_event_loop()
loop = asyncio.get_running_loop()
loop.set_debug(True)
2 years ago
os.makedirs(DATA_DIR, exist_ok=True)
for section in config.sections():
bot_config = config[section]
connection = ChatClient(bot_config['matrix_homeserver'], bot_config['matrix_username'], bot_config['matrix_password'], bot_config.get('matrix_device_name', 'matrix-nio'))
await connection.persist(f"{DATA_DIR}/{section}/matrix")
2 years ago
bot = ChatBot(section, connection)
await bot.persist(f"{DATA_DIR}/{section}")
2 years ago
bot.init_character(
bot_config['persona'],
bot_config['scenario'],
bot_config['greeting'],
json.loads(bot_config.get('example_dialogue', "[]")),
bot_config.get('nsfw', False),
bot_config.get('temperature', 0.72),
)
if config.has_option(section, 'owner'):
bot.owner = config[section]['owner']
# if config.has_option(section, 'translate'):
# bot.translate = config[section]['translate']
# translate.init(bot.translate, "en")
# translate.init("en", bot.translate)
await bot.load_ai(
json.loads(bot_config['available_text_endpoints']),
json.loads(bot_config['available_image_endpoints']),
)
await bot.connect()
bots.append(bot)
webui = WebUI()
2 years ago
await webui.connect(bots)
async def shutdown(signal, loop):
"""Cleanup tasks and shut down"""
logger.info(f"Received exit signal {signal.name} ...")
await webui.stop()
for bot in bots:
await bot.disconnect()
tasks = [t for t in asyncio.all_tasks() if t is not asyncio.current_task()]
[task.cancel() for task in tasks]
logging.info(f"Cancelling {len(tasks)} outstanding tasks")
await asyncio.gather(*tasks, return_exceptions=True)
logging.info(f"Flushing metrics")
loop.stop()
2 years ago
# for signame in {'SIGINT', 'SIGTERM'}:
# loop.add_signal_handler(
# getattr(signal, signame),
# functools.partial(shutdown, signame, loop))
for s in {signal.SIGHUP, signal.SIGTERM, signal.SIGINT}:
loop.add_signal_handler(
s, lambda s=s: asyncio.create_task(shutdown(s, loop)))
try:
2 years ago
2 years ago
if sys.version_info[0] == 3 and sys.version_info[1] < 11:
tasks = []
for bot in bots:
2 years ago
task = asyncio.create_task(bot.connection.sync_forever(timeout=180000, full_state=True)) # 30000
2 years ago
tasks.append(task)
webui.task = asyncio.create_task(webui.run_task())
tasks.append(webui.task)
2 years ago
await asyncio.gather(*tasks)
else:
async with asyncio.TaskGroup() as tg:
for bot in bots:
2 years ago
task = tg.create_task(bot.connection.sync_forever(timeout=180000, full_state=True)) # 30000
webui.task = tg.create_task(webui.run_task())
2 years ago
# except Exception:
# print(traceback.format_exc())
# sys.exit(1)
2 years ago
except (asyncio.CancelledError, KeyboardInterrupt):
print("Received keyboard interrupt.")
# webui.task.cancel()
# for bot in bots:
# await bot.disconnect()
# sys.exit(0)
finally:
pass
#loop.close()
2 years ago
2 years ago
2 years ago
if __name__ == "__main__":
asyncio.run(main())