-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy pathbot.py
79 lines (66 loc) · 2.54 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
import logging
import traceback
from aiogram.client.default import DefaultBotProperties
from aiogram.fsm.storage.redis import RedisStorage
from aiogram.types import BufferedInputFile
from redis.asyncio import Redis
import config
from aiogram import Bot, Dispatcher
from aiogram.enums import ParseMode
from fastapi import FastAPI, Request, status, HTTPException
from config import TOKEN, WEBHOOK_URL, ADMIN_ID_LIST, WEBHOOK_SECRET_TOKEN
from db import create_db_and_tables
import uvicorn
from fastapi.responses import JSONResponse
from services.notification import NotificationService
redis = Redis(password=config.REDIS_PASSWORD)
bot = Bot(TOKEN, default=DefaultBotProperties(parse_mode=ParseMode.HTML))
dp = Dispatcher(storage=RedisStorage(redis))
app = FastAPI()
@app.post(config.WEBHOOK_PATH)
async def webhook(request: Request):
secret_token = request.headers.get("X-Telegram-Bot-Api-Secret-Token")
if secret_token != WEBHOOK_SECRET_TOKEN:
raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED, detail="Unauthorized")
try:
update_data = await request.json()
await dp.feed_webhook_update(bot, update_data)
return {"status": "ok"}
except Exception as e:
logging.error(f"Error processing webhook: {e}")
return {"status": "error"}, status.HTTP_500_INTERNAL_SERVER_ERROR
@app.on_event("startup")
async def on_startup():
await create_db_and_tables()
await bot.set_webhook(
url=WEBHOOK_URL,
secret_token=WEBHOOK_SECRET_TOKEN
)
for admin in ADMIN_ID_LIST:
try:
await bot.send_message(admin, 'Bot is working')
except Exception as e:
logging.warning(e)
@app.on_event("shutdown")
async def on_shutdown():
logging.warning('Shutting down..')
await bot.delete_webhook()
await dp.storage.close()
logging.warning('Bye!')
@app.exception_handler(Exception)
async def exception_handler(request: Request, exc: Exception):
traceback_str = traceback.format_exc()
admin_notification = (
f"Critical error caused by {exc}\n\n"
f"Stack trace:\n{traceback_str}"
)
if len(admin_notification) > 4096:
byte_array = bytearray(admin_notification, 'utf-8')
admin_notification = BufferedInputFile(byte_array, "exception.txt")
await NotificationService.send_to_admins(admin_notification, None)
return JSONResponse(
status_code=500,
content={"message": f"An error occurred: {str(exc)}"},
)
def main() -> None:
uvicorn.run(app, host=config.WEBAPP_HOST, port=config.WEBAPP_PORT)