mirror of
https://github.com/BEDOLAGA-DEV/remnawave-bedolaga-telegram-bot.git
synced 2026-01-19 19:32:10 +00:00
Preserve normalized Stars invoice amounts
This commit is contained in:
@@ -2,7 +2,7 @@ import logging
|
||||
import hashlib
|
||||
import hmac
|
||||
import uuid
|
||||
from decimal import Decimal, InvalidOperation, ROUND_HALF_UP
|
||||
from decimal import Decimal, InvalidOperation, ROUND_HALF_UP, ROUND_FLOOR
|
||||
from typing import Optional, Dict, Any
|
||||
from datetime import datetime
|
||||
from aiogram import Bot
|
||||
@@ -106,16 +106,32 @@ class PaymentService:
|
||||
self,
|
||||
amount_kopeks: int,
|
||||
description: str,
|
||||
payload: Optional[str] = None
|
||||
payload: Optional[str] = None,
|
||||
*,
|
||||
stars_amount: Optional[int] = None,
|
||||
) -> str:
|
||||
|
||||
|
||||
if not self.bot or not self.stars_service:
|
||||
raise ValueError("Bot instance required for Stars payments")
|
||||
|
||||
|
||||
try:
|
||||
amount_rubles = Decimal(amount_kopeks) / Decimal(100)
|
||||
stars_amount = TelegramStarsService.calculate_stars_from_rubles(float(amount_rubles))
|
||||
|
||||
|
||||
if stars_amount is None:
|
||||
rate = Decimal(str(settings.get_stars_rate()))
|
||||
if rate <= 0:
|
||||
raise ValueError("Stars rate must be positive")
|
||||
|
||||
normalized_stars = (amount_rubles / rate).to_integral_value(
|
||||
rounding=ROUND_FLOOR
|
||||
)
|
||||
stars_amount = int(normalized_stars)
|
||||
if stars_amount <= 0:
|
||||
stars_amount = 1
|
||||
|
||||
if stars_amount <= 0:
|
||||
raise ValueError("Stars amount must be positive")
|
||||
|
||||
invoice_link = await self.bot.create_invoice_link(
|
||||
title="Пополнение баланса VPN",
|
||||
description=f"{description} (≈{stars_amount} ⭐)",
|
||||
@@ -828,6 +844,7 @@ class PaymentService:
|
||||
language: str,
|
||||
ttl_seconds: Optional[int] = None,
|
||||
payer_email: Optional[str] = None,
|
||||
payment_method: Optional[str] = None,
|
||||
) -> Optional[Dict[str, Any]]:
|
||||
|
||||
if not self.pal24_service or not self.pal24_service.is_configured:
|
||||
@@ -858,6 +875,8 @@ class PaymentService:
|
||||
"language": language,
|
||||
}
|
||||
|
||||
normalized_payment_method = (payment_method or "SBP").upper()
|
||||
|
||||
try:
|
||||
response = await self.pal24_service.create_bill(
|
||||
amount_kopeks=amount_kopeks,
|
||||
@@ -867,7 +886,7 @@ class PaymentService:
|
||||
ttl_seconds=ttl_seconds,
|
||||
custom_payload=custom_payload,
|
||||
payer_email=payer_email,
|
||||
payment_method="SBP",
|
||||
payment_method=normalized_payment_method,
|
||||
)
|
||||
except Pal24APIError as error:
|
||||
logger.error("Ошибка Pal24 API при создании счета: %s", error)
|
||||
@@ -962,6 +981,7 @@ class PaymentService:
|
||||
"sbp_url": transfer_url or primary_link,
|
||||
"card_url": card_url,
|
||||
"transfer_url": transfer_url,
|
||||
"payment_method": normalized_payment_method,
|
||||
}
|
||||
|
||||
logger.info(
|
||||
|
||||
File diff suppressed because it is too large
Load Diff
@@ -253,6 +253,72 @@ class MiniAppReferralInfo(BaseModel):
|
||||
referrals: Optional[MiniAppReferralList] = None
|
||||
|
||||
|
||||
class MiniAppPaymentMethodsRequest(BaseModel):
|
||||
init_data: str = Field(..., alias="initData")
|
||||
|
||||
|
||||
class MiniAppPaymentMethod(BaseModel):
|
||||
id: str
|
||||
icon: Optional[str] = None
|
||||
requires_amount: bool = False
|
||||
currency: str = "RUB"
|
||||
min_amount_kopeks: Optional[int] = None
|
||||
max_amount_kopeks: Optional[int] = None
|
||||
amount_step_kopeks: Optional[int] = None
|
||||
|
||||
|
||||
class MiniAppPaymentMethodsResponse(BaseModel):
|
||||
methods: List[MiniAppPaymentMethod] = Field(default_factory=list)
|
||||
|
||||
|
||||
class MiniAppPaymentCreateRequest(BaseModel):
|
||||
init_data: str = Field(..., alias="initData")
|
||||
method: str
|
||||
amount_rubles: Optional[float] = Field(default=None, alias="amountRubles")
|
||||
amount_kopeks: Optional[int] = Field(default=None, alias="amountKopeks")
|
||||
payment_option: Optional[str] = Field(default=None, alias="option")
|
||||
|
||||
|
||||
class MiniAppPaymentCreateResponse(BaseModel):
|
||||
success: bool = True
|
||||
method: str
|
||||
payment_url: Optional[str] = None
|
||||
amount_kopeks: Optional[int] = None
|
||||
extra: Dict[str, Any] = Field(default_factory=dict)
|
||||
|
||||
|
||||
class MiniAppPaymentStatusQuery(BaseModel):
|
||||
method: str
|
||||
local_payment_id: Optional[int] = Field(default=None, alias="localPaymentId")
|
||||
invoice_id: Optional[str] = Field(default=None, alias="invoiceId")
|
||||
payment_id: Optional[str] = Field(default=None, alias="paymentId")
|
||||
payload: Optional[str] = None
|
||||
amount_kopeks: Optional[int] = Field(default=None, alias="amountKopeks")
|
||||
started_at: Optional[str] = Field(default=None, alias="startedAt")
|
||||
|
||||
|
||||
class MiniAppPaymentStatusRequest(BaseModel):
|
||||
init_data: str = Field(..., alias="initData")
|
||||
payments: List[MiniAppPaymentStatusQuery] = Field(default_factory=list)
|
||||
|
||||
|
||||
class MiniAppPaymentStatusResult(BaseModel):
|
||||
method: str
|
||||
status: str
|
||||
is_paid: bool = False
|
||||
amount_kopeks: Optional[int] = None
|
||||
currency: Optional[str] = None
|
||||
completed_at: Optional[datetime] = None
|
||||
transaction_id: Optional[int] = None
|
||||
external_id: Optional[str] = None
|
||||
message: Optional[str] = None
|
||||
extra: Dict[str, Any] = Field(default_factory=dict)
|
||||
|
||||
|
||||
class MiniAppPaymentStatusResponse(BaseModel):
|
||||
results: List[MiniAppPaymentStatusResult] = Field(default_factory=list)
|
||||
|
||||
|
||||
class MiniAppSubscriptionResponse(BaseModel):
|
||||
success: bool = True
|
||||
subscription_id: int
|
||||
|
||||
1819
miniapp/index.html
1819
miniapp/index.html
File diff suppressed because it is too large
Load Diff
353
tests/test_miniapp_payments.py
Normal file
353
tests/test_miniapp_payments.py
Normal file
@@ -0,0 +1,353 @@
|
||||
import os
|
||||
import sys
|
||||
import types
|
||||
from datetime import datetime, timedelta
|
||||
from pathlib import Path
|
||||
|
||||
import pytest
|
||||
|
||||
ROOT_DIR = Path(__file__).resolve().parents[1]
|
||||
if str(ROOT_DIR) not in sys.path:
|
||||
sys.path.insert(0, str(ROOT_DIR))
|
||||
|
||||
BACKUP_DIR = ROOT_DIR / 'data' / 'backups'
|
||||
BACKUP_DIR.mkdir(parents=True, exist_ok=True)
|
||||
|
||||
os.environ.setdefault('BOT_TOKEN', 'test-token')
|
||||
|
||||
from app.config import settings
|
||||
from app.webapi.routes import miniapp
|
||||
from app.database.models import PaymentMethod
|
||||
from app.webapi.schemas.miniapp import (
|
||||
MiniAppPaymentCreateRequest,
|
||||
MiniAppPaymentMethodsRequest,
|
||||
MiniAppPaymentStatusQuery,
|
||||
)
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def anyio_backend():
|
||||
return 'asyncio'
|
||||
|
||||
|
||||
def test_compute_cryptobot_limits_scale_with_rate():
|
||||
low_rate_min, low_rate_max = miniapp._compute_cryptobot_limits(70.0)
|
||||
high_rate_min, high_rate_max = miniapp._compute_cryptobot_limits(120.0)
|
||||
|
||||
assert low_rate_min == 7000
|
||||
assert low_rate_max == 7000000
|
||||
assert high_rate_min == 12000
|
||||
assert high_rate_max == 12000000
|
||||
assert high_rate_min > low_rate_min
|
||||
assert high_rate_max > low_rate_max
|
||||
|
||||
|
||||
@pytest.mark.anyio("asyncio")
|
||||
async def test_create_payment_link_pal24_uses_selected_option(monkeypatch):
|
||||
monkeypatch.setattr(settings, 'PAL24_ENABLED', True, raising=False)
|
||||
monkeypatch.setattr(settings, 'PAL24_API_TOKEN', 'token', raising=False)
|
||||
monkeypatch.setattr(settings, 'PAL24_SHOP_ID', 'shop', raising=False)
|
||||
monkeypatch.setattr(settings, 'PAL24_MIN_AMOUNT_KOPEKS', 1000, raising=False)
|
||||
monkeypatch.setattr(settings, 'PAL24_MAX_AMOUNT_KOPEKS', 5000000, raising=False)
|
||||
|
||||
captured_calls = []
|
||||
|
||||
class DummyPaymentService:
|
||||
def __init__(self, *args, **kwargs):
|
||||
pass
|
||||
|
||||
async def create_pal24_payment(self, db, **kwargs):
|
||||
captured_calls.append({'db': db, **kwargs})
|
||||
return {
|
||||
'local_payment_id': 101,
|
||||
'bill_id': 'BILL42',
|
||||
'order_id': 'ORD42',
|
||||
'payment_method': kwargs.get('payment_method'),
|
||||
'sbp_url': 'https://sbp',
|
||||
'card_url': 'https://card',
|
||||
'link_url': 'https://link',
|
||||
}
|
||||
|
||||
async def fake_resolve_user(db, init_data):
|
||||
return types.SimpleNamespace(id=123, language='ru'), {}
|
||||
|
||||
monkeypatch.setattr(miniapp, 'PaymentService', lambda *args, **kwargs: DummyPaymentService())
|
||||
monkeypatch.setattr(miniapp, '_resolve_user_from_init_data', fake_resolve_user)
|
||||
|
||||
payload = MiniAppPaymentCreateRequest(
|
||||
initData='test',
|
||||
method='pal24',
|
||||
amountKopeks=15000,
|
||||
option='card',
|
||||
)
|
||||
|
||||
response = await miniapp.create_payment_link(payload, db=types.SimpleNamespace())
|
||||
|
||||
assert response.payment_url == 'https://card'
|
||||
assert response.extra['selected_option'] == 'card'
|
||||
assert response.extra['payment_method'] == 'CARD'
|
||||
assert captured_calls and captured_calls[0]['payment_method'] == 'CARD'
|
||||
|
||||
|
||||
@pytest.mark.anyio("asyncio")
|
||||
async def test_resolve_yookassa_status_includes_identifiers(monkeypatch):
|
||||
payment = types.SimpleNamespace(
|
||||
id=55,
|
||||
user_id=1,
|
||||
amount_kopeks=15000,
|
||||
currency='RUB',
|
||||
status='pending',
|
||||
is_paid=False,
|
||||
captured_at=None,
|
||||
updated_at=None,
|
||||
created_at=datetime.utcnow(),
|
||||
transaction_id=42,
|
||||
yookassa_payment_id='yk_1',
|
||||
)
|
||||
|
||||
async def fake_get_by_local_id(db, local_id):
|
||||
return payment if local_id == 55 else None
|
||||
|
||||
async def fake_get_by_id(db, payment_id):
|
||||
return None
|
||||
|
||||
stub_module = types.SimpleNamespace(
|
||||
get_yookassa_payment_by_local_id=fake_get_by_local_id,
|
||||
get_yookassa_payment_by_id=fake_get_by_id,
|
||||
)
|
||||
monkeypatch.setitem(sys.modules, 'app.database.crud.yookassa', stub_module)
|
||||
|
||||
user = types.SimpleNamespace(id=1)
|
||||
query = MiniAppPaymentStatusQuery(
|
||||
method='yookassa',
|
||||
localPaymentId=55,
|
||||
paymentId='yk_1',
|
||||
amountKopeks=15000,
|
||||
startedAt='2024-01-01T00:00:00Z',
|
||||
payload='payload123',
|
||||
)
|
||||
|
||||
result = await miniapp._resolve_yookassa_payment_status(db=None, user=user, query=query)
|
||||
|
||||
assert result.extra['local_payment_id'] == 55
|
||||
assert result.extra['payment_id'] == 'yk_1'
|
||||
assert result.extra['invoice_id'] == 'yk_1'
|
||||
assert result.extra['payload'] == 'payload123'
|
||||
assert result.extra['started_at'] == '2024-01-01T00:00:00Z'
|
||||
|
||||
|
||||
@pytest.mark.anyio("asyncio")
|
||||
async def test_resolve_pal24_status_includes_identifiers(monkeypatch):
|
||||
async def fake_get_pal24_payment_by_bill_id(db, bill_id):
|
||||
return None
|
||||
|
||||
stub_module = types.SimpleNamespace(
|
||||
get_pal24_payment_by_bill_id=fake_get_pal24_payment_by_bill_id,
|
||||
)
|
||||
monkeypatch.setitem(sys.modules, 'app.database.crud.pal24', stub_module)
|
||||
|
||||
paid_at = datetime.utcnow()
|
||||
|
||||
payment = types.SimpleNamespace(
|
||||
id=321,
|
||||
user_id=1,
|
||||
amount_kopeks=25000,
|
||||
currency='RUB',
|
||||
is_paid=True,
|
||||
status='PAID',
|
||||
paid_at=paid_at,
|
||||
updated_at=paid_at,
|
||||
created_at=paid_at - timedelta(minutes=1),
|
||||
transaction_id=777,
|
||||
bill_id='BILL99',
|
||||
order_id='ORD99',
|
||||
payment_method='SBP',
|
||||
)
|
||||
|
||||
class StubPal24Service:
|
||||
async def get_pal24_payment_status(self, db, local_id):
|
||||
assert local_id == 321
|
||||
return {
|
||||
'payment': payment,
|
||||
'status': 'PAID',
|
||||
'remote_status': 'PAID',
|
||||
}
|
||||
|
||||
user = types.SimpleNamespace(id=1)
|
||||
query = MiniAppPaymentStatusQuery(
|
||||
method='pal24',
|
||||
localPaymentId=321,
|
||||
amountKopeks=25000,
|
||||
startedAt='2024-01-01T00:00:00Z',
|
||||
payload='pal24_payload',
|
||||
)
|
||||
|
||||
result = await miniapp._resolve_pal24_payment_status(
|
||||
StubPal24Service(),
|
||||
db=None,
|
||||
user=user,
|
||||
query=query,
|
||||
)
|
||||
|
||||
assert result.status == 'paid'
|
||||
assert result.extra['local_payment_id'] == 321
|
||||
assert result.extra['bill_id'] == 'BILL99'
|
||||
assert result.extra['order_id'] == 'ORD99'
|
||||
assert result.extra['payment_method'] == 'SBP'
|
||||
assert result.extra['payload'] == 'pal24_payload'
|
||||
assert result.extra['started_at'] == '2024-01-01T00:00:00Z'
|
||||
assert result.extra['remote_status'] == 'PAID'
|
||||
|
||||
|
||||
@pytest.mark.anyio("asyncio")
|
||||
async def test_create_payment_link_stars_normalizes_amount(monkeypatch):
|
||||
monkeypatch.setattr(settings, 'TELEGRAM_STARS_ENABLED', True, raising=False)
|
||||
monkeypatch.setattr(settings, 'TELEGRAM_STARS_RATE_RUB', 1000.0, raising=False)
|
||||
monkeypatch.setattr(settings, 'BOT_TOKEN', 'test-token', raising=False)
|
||||
|
||||
captured = {}
|
||||
|
||||
class DummyPaymentService:
|
||||
def __init__(self, bot):
|
||||
captured['bot'] = bot
|
||||
|
||||
async def create_stars_invoice(
|
||||
self,
|
||||
amount_kopeks,
|
||||
description,
|
||||
payload,
|
||||
*,
|
||||
stars_amount=None,
|
||||
):
|
||||
captured['amount_kopeks'] = amount_kopeks
|
||||
captured['description'] = description
|
||||
captured['payload'] = payload
|
||||
captured['stars_amount'] = stars_amount
|
||||
return 'https://invoice.example'
|
||||
|
||||
class DummySession:
|
||||
def __init__(self):
|
||||
self.closed = False
|
||||
|
||||
async def close(self):
|
||||
self.closed = True
|
||||
captured['session_closed'] = True
|
||||
|
||||
class DummyBot:
|
||||
def __init__(self, token):
|
||||
captured['bot_token'] = token
|
||||
self.session = DummySession()
|
||||
|
||||
async def fake_resolve_user(db, init_data):
|
||||
return types.SimpleNamespace(id=7, language='ru'), {}
|
||||
|
||||
monkeypatch.setattr(miniapp, 'PaymentService', lambda bot: DummyPaymentService(bot))
|
||||
monkeypatch.setattr(miniapp, 'Bot', DummyBot)
|
||||
monkeypatch.setattr(miniapp, '_resolve_user_from_init_data', fake_resolve_user)
|
||||
|
||||
payload = MiniAppPaymentCreateRequest(
|
||||
initData='data',
|
||||
method='stars',
|
||||
amountKopeks=101000,
|
||||
)
|
||||
|
||||
response = await miniapp.create_payment_link(payload, db=types.SimpleNamespace())
|
||||
|
||||
assert response.payment_url == 'https://invoice.example'
|
||||
assert response.amount_kopeks == 100000
|
||||
assert response.extra['stars_amount'] == 1
|
||||
assert response.extra['requested_amount_kopeks'] == 101000
|
||||
assert captured['amount_kopeks'] == 100000
|
||||
assert captured['stars_amount'] == 1
|
||||
assert captured['bot_token'] == 'test-token'
|
||||
assert captured.get('session_closed') is True
|
||||
|
||||
|
||||
@pytest.mark.anyio("asyncio")
|
||||
async def test_get_payment_methods_exposes_stars_min_amount(monkeypatch):
|
||||
monkeypatch.setattr(settings, 'TELEGRAM_STARS_ENABLED', True, raising=False)
|
||||
monkeypatch.setattr(settings, 'TELEGRAM_STARS_RATE_RUB', 999.99, raising=False)
|
||||
|
||||
async def fake_resolve_user(db, init_data):
|
||||
return types.SimpleNamespace(id=1, language='ru'), {}
|
||||
|
||||
monkeypatch.setattr(miniapp, '_resolve_user_from_init_data', fake_resolve_user)
|
||||
|
||||
payload = MiniAppPaymentMethodsRequest(initData='abc')
|
||||
|
||||
response = await miniapp.get_payment_methods(payload, db=types.SimpleNamespace())
|
||||
|
||||
stars_method = next((method for method in response.methods if method.id == 'stars'), None)
|
||||
assert stars_method is not None
|
||||
assert stars_method.min_amount_kopeks == 99999
|
||||
assert stars_method.amount_step_kopeks == 99999
|
||||
@pytest.mark.anyio("asyncio")
|
||||
async def test_find_recent_deposit_ignores_transactions_before_attempt():
|
||||
started_at = datetime(2024, 5, 1, 12, 0, 0)
|
||||
|
||||
transaction = types.SimpleNamespace(
|
||||
id=10,
|
||||
amount_kopeks=1000,
|
||||
completed_at=None,
|
||||
created_at=started_at - timedelta(minutes=1),
|
||||
)
|
||||
|
||||
class DummyResult:
|
||||
def __init__(self, value):
|
||||
self._value = value
|
||||
|
||||
def scalar_one_or_none(self):
|
||||
return self._value
|
||||
|
||||
class DummySession:
|
||||
def __init__(self, value):
|
||||
self._value = value
|
||||
|
||||
async def execute(self, query): # noqa: ARG002
|
||||
return DummyResult(self._value)
|
||||
|
||||
result = await miniapp._find_recent_deposit(
|
||||
DummySession(transaction),
|
||||
user_id=1,
|
||||
payment_method=PaymentMethod.TELEGRAM_STARS,
|
||||
amount_kopeks=1000,
|
||||
started_at=started_at,
|
||||
)
|
||||
|
||||
assert result is None
|
||||
|
||||
|
||||
@pytest.mark.anyio("asyncio")
|
||||
async def test_find_recent_deposit_accepts_recent_transactions():
|
||||
started_at = datetime(2024, 5, 1, 12, 0, 0)
|
||||
|
||||
transaction = types.SimpleNamespace(
|
||||
id=11,
|
||||
amount_kopeks=1000,
|
||||
completed_at=started_at + timedelta(seconds=5),
|
||||
created_at=started_at + timedelta(seconds=5),
|
||||
)
|
||||
|
||||
class DummyResult:
|
||||
def __init__(self, value):
|
||||
self._value = value
|
||||
|
||||
def scalar_one_or_none(self):
|
||||
return self._value
|
||||
|
||||
class DummySession:
|
||||
def __init__(self, value):
|
||||
self._value = value
|
||||
|
||||
async def execute(self, query): # noqa: ARG002
|
||||
return DummyResult(self._value)
|
||||
|
||||
result = await miniapp._find_recent_deposit(
|
||||
DummySession(transaction),
|
||||
user_id=1,
|
||||
payment_method=PaymentMethod.TELEGRAM_STARS,
|
||||
amount_kopeks=1000,
|
||||
started_at=started_at,
|
||||
)
|
||||
|
||||
assert result is transaction
|
||||
Reference in New Issue
Block a user