Files
remnawave-bedolaga-telegram…/app/external/telegram_stars.py
c0mrade 9a2aea038a chore: add uv package manager and ruff linter configuration
- Add pyproject.toml with uv and ruff configuration
- Pin Python version to 3.13 via .python-version
- Add Makefile commands: lint, format, fix
- Apply ruff formatting to entire codebase
- Remove unused imports (base64 in yookassa/simple_subscription)
- Update .gitignore for new config files
2026-01-24 17:45:27 +03:00

113 lines
3.8 KiB
Python

import logging
from decimal import ROUND_HALF_UP, Decimal
from typing import Any
from aiogram import Bot
from aiogram.types import InlineKeyboardMarkup, LabeledPrice
from app.config import settings
logger = logging.getLogger(__name__)
class TelegramStarsService:
def __init__(self, bot: Bot):
self.bot = bot
@staticmethod
def calculate_stars_from_rubles(rubles: float) -> int:
return settings.rubles_to_stars(rubles)
@staticmethod
def calculate_rubles_from_stars(stars: int) -> Decimal:
rate = Decimal(str(settings.get_stars_rate()))
return (Decimal(stars) * rate).quantize(Decimal('0.01'), rounding=ROUND_HALF_UP)
async def create_invoice(
self,
chat_id: int,
title: str,
description: str,
amount_kopeks: int,
payload: str,
start_parameter: str | None = None,
) -> str | None:
try:
amount_rubles = Decimal(amount_kopeks) / Decimal(100)
stars_amount = self.calculate_stars_from_rubles(float(amount_rubles))
stars_rate = settings.get_stars_rate()
invoice_link = await self.bot.create_invoice_link(
title=title,
description=description,
payload=payload,
provider_token='',
currency='XTR',
prices=[LabeledPrice(label=title, amount=stars_amount)],
start_parameter=start_parameter,
)
logger.info(
f'Создан Stars invoice на {stars_amount} звезд (~{settings.format_price(amount_kopeks)}) '
f'для {chat_id}, курс: {stars_rate}₽/⭐'
)
return invoice_link
except Exception as e:
logger.error(f'Ошибка создания Stars invoice: {e}')
return None
async def send_invoice(
self,
chat_id: int,
title: str,
description: str,
amount_kopeks: int,
payload: str,
keyboard: InlineKeyboardMarkup | None = None,
) -> dict[str, Any] | None:
try:
amount_rubles = Decimal(amount_kopeks) / Decimal(100)
stars_amount = self.calculate_stars_from_rubles(float(amount_rubles))
stars_rate = settings.get_stars_rate()
message = await self.bot.send_invoice(
chat_id=chat_id,
title=title,
description=description,
payload=payload,
provider_token='',
currency='XTR',
prices=[LabeledPrice(label=title, amount=stars_amount)],
reply_markup=keyboard,
)
logger.info(
f'Отправлен Stars invoice {message.message_id} на {stars_amount} звезд '
f'(~{settings.format_price(amount_kopeks)}), курс: {stars_rate}₽/⭐'
)
return {
'message_id': message.message_id,
'stars_amount': stars_amount,
'rubles_amount': float(amount_rubles),
'payload': payload,
}
except Exception as e:
logger.error(f'Ошибка отправки Stars invoice: {e}')
return None
async def answer_pre_checkout_query(
self, pre_checkout_query_id: str, ok: bool = True, error_message: str | None = None
) -> bool:
try:
await self.bot.answer_pre_checkout_query(
pre_checkout_query_id=pre_checkout_query_id, ok=ok, error_message=error_message
)
logger.info(f'Ответ на pre_checkout_query: ok={ok}')
return True
except Exception as e:
logger.error(f'Ошибка ответа на pre_checkout_query: {e}')
return False