-
Notifications
You must be signed in to change notification settings - Fork 20
/
bot.py
135 lines (109 loc) · 4.08 KB
/
bot.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
import asyncio
import os
import sys
from multiprocessing import Process
from typing import Awaitable, Callable
import aiohttp
from bots.types import BotCallbacks, BotConfig, BotParams
from bots.webrtc.bot_error_pipeline import bot_error_pipeline_task
from bots.webrtc.bot_pipeline import bot_pipeline
from bots.webrtc.bot_pipeline_runner import BotPipelineRunner
from common.config import SERVICE_API_KEYS
from common.database import DatabaseSessionFactory
from fastapi import HTTPException, status
from loguru import logger
from sqlalchemy.ext.asyncio import AsyncSession
from pipecat.pipeline.task import PipelineParams, PipelineTask
from pipecat.transports.services.helpers.daily_rest import (
DailyRESTHelper,
DailyRoomParams,
)
MAX_SESSION_TIME = int(os.getenv("BOT_MAX_VOICE_SESSION_TIME", 15 * 60)) or 15 * 60
async def _cleanup(room_url: str, config: BotConfig):
async with aiohttp.ClientSession() as session:
debug_room = os.getenv("USE_DEBUG_ROOM", None)
if debug_room:
return
transport_api_key = SERVICE_API_KEYS["daily"]
helper = DailyRESTHelper(
daily_api_key=str(transport_api_key),
aiohttp_session=session,
)
try:
logger.info(f"Deleting room {room_url}")
await helper.delete_room_by_url(room_url)
except Exception as e:
logger.error(f"Bot failed to delete room: {e}")
async def _pipeline_task(
params: BotParams,
config: BotConfig,
room_url: str,
room_token: str,
db: AsyncSession,
) -> Callable[[BotCallbacks], Awaitable[PipelineTask]]:
async def create_task(callbacks: BotCallbacks) -> PipelineTask:
pipeline = await bot_pipeline(params, config, callbacks, room_url, room_token, db)
task = PipelineTask(
pipeline,
params=PipelineParams(
allow_interruptions=True,
enable_metrics=True,
send_initial_empty_metrics=False,
),
)
return task
return create_task
async def _bot_main(
params: BotParams,
config: BotConfig,
room_url: str,
room_token: str,
):
subprocess_session_factory = DatabaseSessionFactory()
async with subprocess_session_factory() as db:
bot_runner = BotPipelineRunner()
try:
task_creator = await _pipeline_task(params, config, room_url, room_token, db)
await bot_runner.start(task_creator)
except Exception as e:
logger.error(f"Error running bot: {e}")
task_creator = await bot_error_pipeline_task(
room_url, room_token, f"Error running bot: {e}"
)
await bot_runner.start(task_creator)
await _cleanup(room_url, config)
logger.info("Bot has finished. Bye!")
await subprocess_session_factory.engine.dispose()
def _bot_process(
params: BotParams,
config: BotConfig,
room_url: str,
room_token: str,
):
# This is a different process so we need to make sure we have the right log level.
logger.remove()
logger.add(sys.stderr, level=os.getenv("BOT_LOG_LEVEL", "INFO"))
asyncio.run(_bot_main(params, config, room_url, room_token))
async def bot_create(daily_api_key: str):
async with aiohttp.ClientSession() as session:
daily_rest_helper = DailyRESTHelper(
daily_api_key=daily_api_key,
aiohttp_session=session,
)
try:
room = await daily_rest_helper.create_room(params=DailyRoomParams())
bot_token = await daily_rest_helper.get_token(room.url, MAX_SESSION_TIME)
user_token = await daily_rest_helper.get_token(room.url, MAX_SESSION_TIME)
except Exception as e:
raise HTTPException(
status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, detail=f"Unable to run bot: {e}"
)
return room, user_token, bot_token
def bot_launch(
params: BotParams,
config: BotConfig,
room_url: str,
room_token: str,
):
process = Process(target=_bot_process, args=(params, config, room_url, room_token))
process.start()