Files
remnawave-bedolaga-telegram…/tests/test_miniapp_payments.py

588 lines
19 KiB
Python

import os
import sys
import types
from datetime import datetime, timedelta
from pathlib import Path
from typing import Any
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_create_payment_link_wata_returns_payload(monkeypatch):
monkeypatch.setattr(settings, 'WATA_ENABLED', True, raising=False)
monkeypatch.setattr(settings, 'WATA_ACCESS_TOKEN', 'token', raising=False)
monkeypatch.setattr(settings, 'WATA_TERMINAL_PUBLIC_ID', 'terminal', raising=False)
monkeypatch.setattr(settings, 'WATA_MIN_AMOUNT_KOPEKS', 1000, raising=False)
monkeypatch.setattr(settings, 'WATA_MAX_AMOUNT_KOPEKS', 5000000, raising=False)
captured_call: dict[str, Any] = {}
class DummyPaymentService:
def __init__(self, *args, **kwargs):
pass
async def create_wata_payment(self, db, **kwargs):
captured_call.update({'db': db, **kwargs})
return {
'local_payment_id': 202,
'payment_link_id': 'link_202',
'payment_url': 'https://wata.example/pay',
'status': 'Opened',
'order_id': 'order_202',
}
async def fake_resolve_user(db, init_data):
return types.SimpleNamespace(id=555, language='ru'), {}
monkeypatch.setattr(miniapp, 'PaymentService', lambda *args, **kwargs: DummyPaymentService())
monkeypatch.setattr(miniapp, '_resolve_user_from_init_data', fake_resolve_user)
payload = MiniAppPaymentCreateRequest(
initData='init',
method='wata',
amountKopeks=25000,
)
response = await miniapp.create_payment_link(payload, db=types.SimpleNamespace())
assert response.payment_url == 'https://wata.example/pay'
assert response.amount_kopeks == 25000
assert response.extra['local_payment_id'] == 202
assert response.extra['payment_link_id'] == 'link_202'
assert response.extra['payment_id'] == 'link_202'
assert response.extra['order_id'] == 'order_202'
assert 'requested_at' in response.extra
assert captured_call.get('user_id') == 555
assert captured_call.get('amount_kopeks') == 25000
assert captured_call.get('description')
@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_payment_status_supports_yookassa_sbp(monkeypatch):
payment = types.SimpleNamespace(
id=77,
user_id=5,
amount_kopeks=25000,
currency='RUB',
status='pending',
is_paid=False,
captured_at=None,
updated_at=None,
created_at=datetime.utcnow(),
transaction_id=None,
yookassa_payment_id='yk_sbp_1',
)
async def fake_get_by_local_id(db, local_id): # noqa: ARG001
return payment if local_id == 77 else None
async def fake_get_by_id(db, payment_id): # noqa: ARG001
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=5)
query = MiniAppPaymentStatusQuery(
method='yookassa_sbp',
localPaymentId=77,
amountKopeks=25000,
startedAt='2024-05-01T10:00:00Z',
payload='sbp_payload',
)
result = await miniapp._resolve_payment_status_entry(
payment_service=types.SimpleNamespace(),
db=None,
user=user,
query=query,
)
assert result.method == 'yookassa_sbp'
assert result.status == 'pending'
assert result.extra['local_payment_id'] == 77
assert result.extra['payment_id'] == 'yk_sbp_1'
assert result.extra['payload'] == 'sbp_payload'
assert result.extra['started_at'] == '2024-05-01T10: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_resolve_wata_payment_status_success():
paid_at = datetime.utcnow()
payment = types.SimpleNamespace(
id=404,
user_id=9,
amount_kopeks=30000,
currency='RUB',
status='Paid',
is_paid=True,
payment_link_id='wata_link_404',
order_id='order_404',
transaction_id=909,
paid_at=paid_at,
updated_at=paid_at,
created_at=paid_at - timedelta(minutes=1),
)
class StubWataService:
async def get_wata_payment_status(self, db, local_payment_id): # noqa: ARG002
assert local_payment_id == 404
return {
'payment': payment,
'status': 'Paid',
'is_paid': True,
'remote_link': {'status': 'Paid'},
'transaction': {'id': 'tx_404', 'status': 'Paid'},
}
query = MiniAppPaymentStatusQuery(
method='wata',
localPaymentId=404,
amountKopeks=30000,
startedAt='2024-06-01T12:00:00Z',
payload='wata_payload',
)
result = await miniapp._resolve_wata_payment_status(
StubWataService(),
db=None,
user=types.SimpleNamespace(id=9),
query=query,
)
assert result.status == 'paid'
assert result.external_id == 'wata_link_404'
assert result.extra['payment_link_id'] == 'wata_link_404'
assert result.extra['transaction']['id'] == 'tx_404'
assert result.extra['payload'] == 'wata_payload'
assert result.extra['started_at'] == '2024-06-01T12:00:00Z'
@pytest.mark.anyio("asyncio")
async def test_resolve_wata_payment_status_uses_payment_link_lookup(monkeypatch):
created_at = datetime.utcnow()
payment = types.SimpleNamespace(
id=505,
user_id=7,
amount_kopeks=15000,
currency='RUB',
status='Opened',
is_paid=False,
payment_link_id='wata_lookup',
order_id='order_lookup',
transaction_id=None,
paid_at=None,
updated_at=None,
created_at=created_at,
)
async def fake_get_wata_payment_by_link_id(db, link_id): # noqa: ARG001
assert link_id == 'wata_lookup'
return payment
monkeypatch.setattr(miniapp, 'get_wata_payment_by_link_id', fake_get_wata_payment_by_link_id)
class StubWataService:
async def get_wata_payment_status(self, db, local_payment_id): # noqa: ARG002
assert local_payment_id == 505
return {
'payment': payment,
'status': payment.status,
'is_paid': False,
'remote_link': None,
'transaction': None,
}
query = MiniAppPaymentStatusQuery(
method='wata',
paymentLinkId='wata_lookup',
amountKopeks=15000,
)
result = await miniapp._resolve_wata_payment_status(
StubWataService(),
db=None,
user=types.SimpleNamespace(id=7),
query=query,
)
assert result.status == 'pending'
assert result.extra['local_payment_id'] == 505
assert result.extra['payment_link_id'] == 'wata_lookup'
assert 'transaction' not in result.extra
@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_get_payment_methods_includes_wata(monkeypatch):
monkeypatch.setattr(settings, 'WATA_ENABLED', True, raising=False)
monkeypatch.setattr(settings, 'WATA_ACCESS_TOKEN', 'token', raising=False)
monkeypatch.setattr(settings, 'WATA_TERMINAL_PUBLIC_ID', 'terminal', raising=False)
monkeypatch.setattr(settings, 'WATA_MIN_AMOUNT_KOPEKS', 5000, raising=False)
monkeypatch.setattr(settings, 'WATA_MAX_AMOUNT_KOPEKS', 7500000, 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())
wata_method = next((method for method in response.methods if method.id == 'wata'), None)
assert wata_method is not None
assert wata_method.min_amount_kopeks == 5000
assert wata_method.max_amount_kopeks == 7500000
assert wata_method.icon == '🌊'
@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