2025-12-16 20:06:16 +07:00
|
|
|
|
import logging
|
|
|
|
|
|
from typing import List
|
|
|
|
|
|
|
|
|
|
|
|
import httpx
|
|
|
|
|
|
from sqlalchemy import select
|
|
|
|
|
|
from sqlalchemy.ext.asyncio import AsyncSession
|
|
|
|
|
|
|
|
|
|
|
|
from app.core.config import settings
|
|
|
|
|
|
from app.models import User, Participant, Marathon
|
|
|
|
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class TelegramNotifier:
|
|
|
|
|
|
"""Service for sending Telegram notifications."""
|
|
|
|
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
|
|
self.bot_token = settings.TELEGRAM_BOT_TOKEN
|
|
|
|
|
|
self.api_url = f"https://api.telegram.org/bot{self.bot_token}"
|
|
|
|
|
|
|
|
|
|
|
|
async def send_message(
|
|
|
|
|
|
self,
|
|
|
|
|
|
chat_id: int,
|
|
|
|
|
|
text: str,
|
2025-12-16 22:43:03 +07:00
|
|
|
|
parse_mode: str = "HTML",
|
|
|
|
|
|
reply_markup: dict | None = None
|
2025-12-16 20:06:16 +07:00
|
|
|
|
) -> bool:
|
|
|
|
|
|
"""Send a message to a Telegram chat."""
|
|
|
|
|
|
if not self.bot_token:
|
|
|
|
|
|
logger.warning("Telegram bot token not configured")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
async with httpx.AsyncClient() as client:
|
2025-12-16 22:43:03 +07:00
|
|
|
|
payload = {
|
|
|
|
|
|
"chat_id": chat_id,
|
|
|
|
|
|
"text": text,
|
|
|
|
|
|
"parse_mode": parse_mode
|
|
|
|
|
|
}
|
|
|
|
|
|
if reply_markup:
|
|
|
|
|
|
payload["reply_markup"] = reply_markup
|
|
|
|
|
|
|
2025-12-16 20:06:16 +07:00
|
|
|
|
response = await client.post(
|
|
|
|
|
|
f"{self.api_url}/sendMessage",
|
2025-12-16 22:43:03 +07:00
|
|
|
|
json=payload,
|
2025-12-16 20:06:16 +07:00
|
|
|
|
timeout=10.0
|
|
|
|
|
|
)
|
|
|
|
|
|
if response.status_code == 200:
|
|
|
|
|
|
return True
|
|
|
|
|
|
else:
|
|
|
|
|
|
logger.error(f"Failed to send message: {response.text}")
|
|
|
|
|
|
return False
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
logger.error(f"Error sending Telegram message: {e}")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
async def notify_user(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
user_id: int,
|
2025-12-16 22:43:03 +07:00
|
|
|
|
message: str,
|
|
|
|
|
|
reply_markup: dict | None = None
|
2025-12-16 20:06:16 +07:00
|
|
|
|
) -> bool:
|
|
|
|
|
|
"""Send notification to a user by user_id."""
|
|
|
|
|
|
result = await db.execute(
|
|
|
|
|
|
select(User).where(User.id == user_id)
|
|
|
|
|
|
)
|
|
|
|
|
|
user = result.scalar_one_or_none()
|
|
|
|
|
|
|
2025-12-16 22:43:03 +07:00
|
|
|
|
if not user:
|
|
|
|
|
|
logger.warning(f"[Notify] User {user_id} not found")
|
2025-12-16 20:06:16 +07:00
|
|
|
|
return False
|
|
|
|
|
|
|
2025-12-16 22:43:03 +07:00
|
|
|
|
if not user.telegram_id:
|
|
|
|
|
|
logger.warning(f"[Notify] User {user_id} ({user.nickname}) has no telegram_id")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
logger.info(f"[Notify] Sending to user {user.nickname} (telegram_id={user.telegram_id})")
|
|
|
|
|
|
return await self.send_message(user.telegram_id, message, reply_markup=reply_markup)
|
2025-12-16 20:06:16 +07:00
|
|
|
|
|
|
|
|
|
|
async def notify_marathon_participants(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
marathon_id: int,
|
|
|
|
|
|
message: str,
|
2026-01-04 02:47:38 +07:00
|
|
|
|
exclude_user_id: int | None = None,
|
|
|
|
|
|
check_setting: str | None = None
|
2025-12-16 20:06:16 +07:00
|
|
|
|
) -> int:
|
2026-01-04 02:47:38 +07:00
|
|
|
|
"""Send notification to all marathon participants with linked Telegram.
|
|
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
|
check_setting: If provided, only send to users with this setting enabled.
|
|
|
|
|
|
Options: 'notify_events', 'notify_disputes', 'notify_moderation'
|
|
|
|
|
|
"""
|
2025-12-16 20:06:16 +07:00
|
|
|
|
result = await db.execute(
|
|
|
|
|
|
select(User)
|
|
|
|
|
|
.join(Participant, Participant.user_id == User.id)
|
|
|
|
|
|
.where(
|
|
|
|
|
|
Participant.marathon_id == marathon_id,
|
|
|
|
|
|
User.telegram_id.isnot(None)
|
|
|
|
|
|
)
|
|
|
|
|
|
)
|
|
|
|
|
|
users = result.scalars().all()
|
|
|
|
|
|
|
|
|
|
|
|
sent_count = 0
|
|
|
|
|
|
for user in users:
|
|
|
|
|
|
if exclude_user_id and user.id == exclude_user_id:
|
|
|
|
|
|
continue
|
2026-01-04 02:47:38 +07:00
|
|
|
|
# Check notification setting if specified
|
|
|
|
|
|
if check_setting and not getattr(user, check_setting, True):
|
|
|
|
|
|
logger.info(f"[Notify] Skipping user {user.nickname} - {check_setting} is disabled")
|
|
|
|
|
|
continue
|
2025-12-16 20:06:16 +07:00
|
|
|
|
if await self.send_message(user.telegram_id, message):
|
|
|
|
|
|
sent_count += 1
|
|
|
|
|
|
|
|
|
|
|
|
return sent_count
|
|
|
|
|
|
|
|
|
|
|
|
# Notification templates
|
|
|
|
|
|
async def notify_event_start(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
marathon_id: int,
|
|
|
|
|
|
event_type: str,
|
|
|
|
|
|
marathon_title: str
|
|
|
|
|
|
) -> int:
|
2026-01-04 02:47:38 +07:00
|
|
|
|
"""Notify participants about event start (respects notify_events setting)."""
|
2025-12-16 20:06:16 +07:00
|
|
|
|
event_messages = {
|
|
|
|
|
|
"golden_hour": f"🌟 <b>Начался Golden Hour</b> в «{marathon_title}»!\n\nВсе очки x1.5 в течение часа!",
|
|
|
|
|
|
"jackpot": f"🎰 <b>JACKPOT</b> в «{marathon_title}»!\n\nОчки x3 за следующий сложный челлендж!",
|
|
|
|
|
|
"double_risk": f"⚡ <b>Double Risk</b> в «{marathon_title}»!\n\nПоловина очков, но дропы бесплатны!",
|
|
|
|
|
|
"common_enemy": f"👥 <b>Common Enemy</b> в «{marathon_title}»!\n\nВсе получают одинаковый челлендж. Первые 3 — бонус!",
|
|
|
|
|
|
"swap": f"🔄 <b>Swap</b> в «{marathon_title}»!\n\nМожно поменяться заданием с другим участником!",
|
|
|
|
|
|
"game_choice": f"🎲 <b>Выбор игры</b> в «{marathon_title}»!\n\nВыбери игру и один из 3 челленджей!"
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
message = event_messages.get(
|
|
|
|
|
|
event_type,
|
|
|
|
|
|
f"📌 Новое событие в «{marathon_title}»!"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2026-01-04 02:47:38 +07:00
|
|
|
|
return await self.notify_marathon_participants(
|
|
|
|
|
|
db, marathon_id, message, check_setting='notify_events'
|
|
|
|
|
|
)
|
2025-12-16 20:06:16 +07:00
|
|
|
|
|
|
|
|
|
|
async def notify_event_end(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
marathon_id: int,
|
|
|
|
|
|
event_type: str,
|
|
|
|
|
|
marathon_title: str
|
|
|
|
|
|
) -> int:
|
2026-01-04 02:47:38 +07:00
|
|
|
|
"""Notify participants about event end (respects notify_events setting)."""
|
2025-12-16 20:06:16 +07:00
|
|
|
|
event_names = {
|
|
|
|
|
|
"golden_hour": "Golden Hour",
|
|
|
|
|
|
"jackpot": "Jackpot",
|
|
|
|
|
|
"double_risk": "Double Risk",
|
|
|
|
|
|
"common_enemy": "Common Enemy",
|
|
|
|
|
|
"swap": "Swap",
|
|
|
|
|
|
"game_choice": "Выбор игры"
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
event_name = event_names.get(event_type, "Событие")
|
|
|
|
|
|
message = f"⏰ <b>{event_name}</b> в «{marathon_title}» завершён"
|
|
|
|
|
|
|
2026-01-04 02:47:38 +07:00
|
|
|
|
return await self.notify_marathon_participants(
|
|
|
|
|
|
db, marathon_id, message, check_setting='notify_events'
|
|
|
|
|
|
)
|
2025-12-16 20:06:16 +07:00
|
|
|
|
|
|
|
|
|
|
async def notify_marathon_start(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
marathon_id: int,
|
|
|
|
|
|
marathon_title: str
|
|
|
|
|
|
) -> int:
|
|
|
|
|
|
"""Notify participants about marathon start."""
|
|
|
|
|
|
message = (
|
|
|
|
|
|
f"🚀 <b>Марафон «{marathon_title}» начался!</b>\n\n"
|
|
|
|
|
|
f"Время крутить колесо и получить первое задание!"
|
|
|
|
|
|
)
|
|
|
|
|
|
return await self.notify_marathon_participants(db, marathon_id, message)
|
|
|
|
|
|
|
|
|
|
|
|
async def notify_marathon_finish(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
marathon_id: int,
|
|
|
|
|
|
marathon_title: str
|
|
|
|
|
|
) -> int:
|
|
|
|
|
|
"""Notify participants about marathon finish."""
|
|
|
|
|
|
message = (
|
|
|
|
|
|
f"🏆 <b>Марафон «{marathon_title}» завершён!</b>\n\n"
|
|
|
|
|
|
f"Зайди на сайт, чтобы увидеть итоговую таблицу!"
|
|
|
|
|
|
)
|
|
|
|
|
|
return await self.notify_marathon_participants(db, marathon_id, message)
|
|
|
|
|
|
|
|
|
|
|
|
async def notify_dispute_raised(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
user_id: int,
|
|
|
|
|
|
marathon_title: str,
|
2025-12-16 22:43:03 +07:00
|
|
|
|
challenge_title: str,
|
|
|
|
|
|
assignment_id: int
|
2025-12-16 20:06:16 +07:00
|
|
|
|
) -> bool:
|
2026-01-04 02:47:38 +07:00
|
|
|
|
"""Notify user about dispute raised on their assignment (respects notify_disputes setting)."""
|
|
|
|
|
|
# Check user's notification settings
|
|
|
|
|
|
result = await db.execute(select(User).where(User.id == user_id))
|
|
|
|
|
|
user = result.scalar_one_or_none()
|
|
|
|
|
|
if user and not user.notify_disputes:
|
|
|
|
|
|
logger.info(f"[Dispute] Skipping user {user.nickname} - notify_disputes is disabled")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
2025-12-16 22:43:03 +07:00
|
|
|
|
logger.info(f"[Dispute] Sending notification to user_id={user_id} for assignment_id={assignment_id}")
|
|
|
|
|
|
|
|
|
|
|
|
dispute_url = f"{settings.FRONTEND_URL}/assignments/{assignment_id}"
|
|
|
|
|
|
logger.info(f"[Dispute] URL: {dispute_url}")
|
|
|
|
|
|
|
|
|
|
|
|
# Telegram requires HTTPS for inline keyboard URLs
|
|
|
|
|
|
use_inline_button = dispute_url.startswith("https://")
|
|
|
|
|
|
|
|
|
|
|
|
if use_inline_button:
|
|
|
|
|
|
message = (
|
|
|
|
|
|
f"⚠️ <b>На твоё задание подан спор</b>\n\n"
|
|
|
|
|
|
f"Марафон: {marathon_title}\n"
|
|
|
|
|
|
f"Задание: {challenge_title}"
|
|
|
|
|
|
)
|
|
|
|
|
|
reply_markup = {
|
|
|
|
|
|
"inline_keyboard": [[
|
|
|
|
|
|
{"text": "Открыть спор", "url": dispute_url}
|
|
|
|
|
|
]]
|
|
|
|
|
|
}
|
|
|
|
|
|
else:
|
|
|
|
|
|
message = (
|
|
|
|
|
|
f"⚠️ <b>На твоё задание подан спор</b>\n\n"
|
|
|
|
|
|
f"Марафон: {marathon_title}\n"
|
|
|
|
|
|
f"Задание: {challenge_title}\n\n"
|
|
|
|
|
|
f"🔗 {dispute_url}"
|
|
|
|
|
|
)
|
|
|
|
|
|
reply_markup = None
|
|
|
|
|
|
|
|
|
|
|
|
result = await self.notify_user(db, user_id, message, reply_markup=reply_markup)
|
|
|
|
|
|
logger.info(f"[Dispute] Notification result: {result}")
|
|
|
|
|
|
return result
|
2025-12-16 20:06:16 +07:00
|
|
|
|
|
|
|
|
|
|
async def notify_dispute_resolved(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
user_id: int,
|
|
|
|
|
|
marathon_title: str,
|
|
|
|
|
|
challenge_title: str,
|
|
|
|
|
|
is_valid: bool
|
|
|
|
|
|
) -> bool:
|
2026-01-04 02:47:38 +07:00
|
|
|
|
"""Notify user about dispute resolution (respects notify_disputes setting)."""
|
|
|
|
|
|
# Check user's notification settings
|
|
|
|
|
|
result = await db.execute(select(User).where(User.id == user_id))
|
|
|
|
|
|
user = result.scalar_one_or_none()
|
|
|
|
|
|
if user and not user.notify_disputes:
|
|
|
|
|
|
logger.info(f"[Dispute] Skipping user {user.nickname} - notify_disputes is disabled")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
2025-12-16 20:06:16 +07:00
|
|
|
|
if is_valid:
|
|
|
|
|
|
message = (
|
|
|
|
|
|
f"❌ <b>Спор признан обоснованным</b>\n\n"
|
|
|
|
|
|
f"Марафон: {marathon_title}\n"
|
|
|
|
|
|
f"Задание: {challenge_title}\n\n"
|
|
|
|
|
|
f"Задание возвращено. Выполни его заново."
|
|
|
|
|
|
)
|
|
|
|
|
|
else:
|
|
|
|
|
|
message = (
|
|
|
|
|
|
f"✅ <b>Спор отклонён</b>\n\n"
|
|
|
|
|
|
f"Марафон: {marathon_title}\n"
|
|
|
|
|
|
f"Задание: {challenge_title}\n\n"
|
|
|
|
|
|
f"Твоё выполнение засчитано!"
|
|
|
|
|
|
)
|
|
|
|
|
|
return await self.notify_user(db, user_id, message)
|
|
|
|
|
|
|
2025-12-17 19:50:55 +07:00
|
|
|
|
async def notify_game_approved(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
user_id: int,
|
|
|
|
|
|
marathon_title: str,
|
|
|
|
|
|
game_title: str
|
|
|
|
|
|
) -> bool:
|
2026-01-04 02:47:38 +07:00
|
|
|
|
"""Notify user that their proposed game was approved (respects notify_moderation setting)."""
|
|
|
|
|
|
# Check user's notification settings
|
|
|
|
|
|
result = await db.execute(select(User).where(User.id == user_id))
|
|
|
|
|
|
user = result.scalar_one_or_none()
|
|
|
|
|
|
if user and not user.notify_moderation:
|
|
|
|
|
|
logger.info(f"[Moderation] Skipping user {user.nickname} - notify_moderation is disabled")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
2025-12-17 19:50:55 +07:00
|
|
|
|
message = (
|
|
|
|
|
|
f"✅ <b>Твоя игра одобрена!</b>\n\n"
|
|
|
|
|
|
f"Марафон: {marathon_title}\n"
|
|
|
|
|
|
f"Игра: {game_title}\n\n"
|
|
|
|
|
|
f"Теперь она доступна для всех участников."
|
|
|
|
|
|
)
|
|
|
|
|
|
return await self.notify_user(db, user_id, message)
|
|
|
|
|
|
|
|
|
|
|
|
async def notify_game_rejected(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
user_id: int,
|
|
|
|
|
|
marathon_title: str,
|
|
|
|
|
|
game_title: str
|
|
|
|
|
|
) -> bool:
|
2026-01-04 02:47:38 +07:00
|
|
|
|
"""Notify user that their proposed game was rejected (respects notify_moderation setting)."""
|
|
|
|
|
|
# Check user's notification settings
|
|
|
|
|
|
result = await db.execute(select(User).where(User.id == user_id))
|
|
|
|
|
|
user = result.scalar_one_or_none()
|
|
|
|
|
|
if user and not user.notify_moderation:
|
|
|
|
|
|
logger.info(f"[Moderation] Skipping user {user.nickname} - notify_moderation is disabled")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
2025-12-17 19:50:55 +07:00
|
|
|
|
message = (
|
|
|
|
|
|
f"❌ <b>Твоя игра отклонена</b>\n\n"
|
|
|
|
|
|
f"Марафон: {marathon_title}\n"
|
|
|
|
|
|
f"Игра: {game_title}\n\n"
|
|
|
|
|
|
f"Ты можешь предложить другую игру."
|
|
|
|
|
|
)
|
|
|
|
|
|
return await self.notify_user(db, user_id, message)
|
|
|
|
|
|
|
2025-12-18 23:47:11 +07:00
|
|
|
|
async def notify_challenge_approved(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
user_id: int,
|
|
|
|
|
|
marathon_title: str,
|
|
|
|
|
|
game_title: str,
|
|
|
|
|
|
challenge_title: str
|
|
|
|
|
|
) -> bool:
|
2026-01-04 02:47:38 +07:00
|
|
|
|
"""Notify user that their proposed challenge was approved (respects notify_moderation setting)."""
|
|
|
|
|
|
# Check user's notification settings
|
|
|
|
|
|
result = await db.execute(select(User).where(User.id == user_id))
|
|
|
|
|
|
user = result.scalar_one_or_none()
|
|
|
|
|
|
if user and not user.notify_moderation:
|
|
|
|
|
|
logger.info(f"[Moderation] Skipping user {user.nickname} - notify_moderation is disabled")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
2025-12-18 23:47:11 +07:00
|
|
|
|
message = (
|
|
|
|
|
|
f"✅ <b>Твой челлендж одобрен!</b>\n\n"
|
|
|
|
|
|
f"Марафон: {marathon_title}\n"
|
|
|
|
|
|
f"Игра: {game_title}\n"
|
|
|
|
|
|
f"Задание: {challenge_title}\n\n"
|
|
|
|
|
|
f"Теперь оно доступно для всех участников."
|
|
|
|
|
|
)
|
|
|
|
|
|
return await self.notify_user(db, user_id, message)
|
|
|
|
|
|
|
|
|
|
|
|
async def notify_challenge_rejected(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
user_id: int,
|
|
|
|
|
|
marathon_title: str,
|
|
|
|
|
|
game_title: str,
|
|
|
|
|
|
challenge_title: str
|
|
|
|
|
|
) -> bool:
|
2026-01-04 02:47:38 +07:00
|
|
|
|
"""Notify user that their proposed challenge was rejected (respects notify_moderation setting)."""
|
|
|
|
|
|
# Check user's notification settings
|
|
|
|
|
|
result = await db.execute(select(User).where(User.id == user_id))
|
|
|
|
|
|
user = result.scalar_one_or_none()
|
|
|
|
|
|
if user and not user.notify_moderation:
|
|
|
|
|
|
logger.info(f"[Moderation] Skipping user {user.nickname} - notify_moderation is disabled")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
2025-12-18 23:47:11 +07:00
|
|
|
|
message = (
|
|
|
|
|
|
f"❌ <b>Твой челлендж отклонён</b>\n\n"
|
|
|
|
|
|
f"Марафон: {marathon_title}\n"
|
|
|
|
|
|
f"Игра: {game_title}\n"
|
|
|
|
|
|
f"Задание: {challenge_title}\n\n"
|
|
|
|
|
|
f"Ты можешь предложить другой челлендж."
|
|
|
|
|
|
)
|
|
|
|
|
|
return await self.notify_user(db, user_id, message)
|
|
|
|
|
|
|
2025-12-29 22:23:34 +03:00
|
|
|
|
async def notify_admin_disputes_pending(
|
|
|
|
|
|
self,
|
|
|
|
|
|
db: AsyncSession,
|
|
|
|
|
|
count: int
|
|
|
|
|
|
) -> bool:
|
|
|
|
|
|
"""Notify admin about disputes waiting for decision."""
|
|
|
|
|
|
if not settings.TELEGRAM_ADMIN_ID:
|
|
|
|
|
|
logger.warning("[Notify] No TELEGRAM_ADMIN_ID configured")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
admin_url = f"{settings.FRONTEND_URL}/admin/disputes"
|
|
|
|
|
|
use_inline_button = admin_url.startswith("https://")
|
|
|
|
|
|
|
|
|
|
|
|
if use_inline_button:
|
|
|
|
|
|
message = (
|
|
|
|
|
|
f"⚠️ <b>{count} оспаривани{'е' if count == 1 else 'й'} ожида{'ет' if count == 1 else 'ют'} решения</b>\n\n"
|
|
|
|
|
|
f"Голосование завершено, требуется ваше решение."
|
|
|
|
|
|
)
|
|
|
|
|
|
reply_markup = {
|
|
|
|
|
|
"inline_keyboard": [[
|
|
|
|
|
|
{"text": "Открыть оспаривания", "url": admin_url}
|
|
|
|
|
|
]]
|
|
|
|
|
|
}
|
|
|
|
|
|
else:
|
|
|
|
|
|
message = (
|
|
|
|
|
|
f"⚠️ <b>{count} оспаривани{'е' if count == 1 else 'й'} ожида{'ет' if count == 1 else 'ют'} решения</b>\n\n"
|
|
|
|
|
|
f"Голосование завершено, требуется ваше решение.\n\n"
|
|
|
|
|
|
f"🔗 {admin_url}"
|
|
|
|
|
|
)
|
|
|
|
|
|
reply_markup = None
|
|
|
|
|
|
|
|
|
|
|
|
return await self.send_message(
|
|
|
|
|
|
int(settings.TELEGRAM_ADMIN_ID),
|
|
|
|
|
|
message,
|
|
|
|
|
|
reply_markup=reply_markup
|
|
|
|
|
|
)
|
|
|
|
|
|
|
2025-12-16 20:06:16 +07:00
|
|
|
|
|
|
|
|
|
|
# Global instance
|
|
|
|
|
|
telegram_notifier = TelegramNotifier()
|