154 lines
6.8 KiB
Python
154 lines
6.8 KiB
Python
from typing import List
|
||
from fastapi import APIRouter, HTTPException, Depends
|
||
from pydantic import BaseModel
|
||
from app.services import DatabaseManager
|
||
from instance.configdb import get_database_manager
|
||
from uuid import UUID
|
||
import logging
|
||
from sqlalchemy.exc import SQLAlchemyError
|
||
|
||
logging.basicConfig(level=logging.INFO)
|
||
logger = logging.getLogger(__name__)
|
||
|
||
router = APIRouter()
|
||
|
||
|
||
class BuySubscriptionRequest(BaseModel):
|
||
telegram_id: int
|
||
plan_id: str
|
||
|
||
class SubscriptionResponse(BaseModel):
|
||
id: str
|
||
plan: str
|
||
vpn_server_id: str
|
||
expiry_date: str
|
||
created_at: str
|
||
updated_at: str
|
||
|
||
# Эндпоинт для покупки подписки
|
||
@router.post("/subscription/buy", response_model=dict)
|
||
async def buy_subscription(
|
||
request_data: BuySubscriptionRequest,
|
||
database_manager: DatabaseManager = Depends(get_database_manager)
|
||
):
|
||
"""
|
||
Покупка подписки.
|
||
"""
|
||
try:
|
||
result = await database_manager.buy_sub(request_data.telegram_id, request_data.plan_id)
|
||
|
||
if result == "ERROR":
|
||
raise HTTPException(status_code=500, detail="ERROR")
|
||
elif result == "INSUFFICIENT_FUNDS":
|
||
raise HTTPException(status_code=400, detail="INSUFFICIENT_FUNDS")
|
||
elif result == "TARIFF_NOT_FOUND":
|
||
raise HTTPException(status_code=400, detail="TARIFF_NOT_FOUND")
|
||
elif result == "ACTIVE_SUBSCRIPTION_EXISTS":
|
||
raise HTTPException(status_code=400, detail="ACTIVE_SUBSCRIPTION_EXISTS")
|
||
result = await database_manager.generate_uri(request_data.telegram_id)
|
||
return {"message": result}
|
||
except HTTPException as http_exc:
|
||
# Пропускаем HTTPException, чтобы FastAPI обработал его корректно
|
||
raise http_exc
|
||
except Exception as e:
|
||
# Обрабатываем остальные исключения
|
||
raise HTTPException(status_code=500, detail=f"Unexpected error: {str(e)}")
|
||
|
||
|
||
# Эндпоинт для получения последней подписки
|
||
@router.get("/subscription/{telegram_id}/last", response_model=SubscriptionResponse)
|
||
async def last_subscription(telegram_id: int, database_manager: DatabaseManager = Depends(get_database_manager)):
|
||
"""
|
||
Возвращает последнюю подписку пользователя.
|
||
"""
|
||
logger.info(f"Получение последней подписки для пользователя: {telegram_id}")
|
||
try:
|
||
subscriptions = await database_manager.get_last_subscriptions(telegram_id=telegram_id, limit=1)
|
||
|
||
if not subscriptions:
|
||
logger.warning(f"Подписки для пользователя {telegram_id} не найдены")
|
||
raise HTTPException(status_code=404, detail="No subscriptions found")
|
||
|
||
sub = subscriptions[0]
|
||
|
||
return {
|
||
"id": sub.id,
|
||
"plan": sub.plan,
|
||
"vpn_server_id": sub.vpn_server_id,
|
||
"expiry_date": sub.expiry_date.isoformat(),
|
||
"created_at": sub.created_at.isoformat(),
|
||
"updated_at": sub.updated_at.isoformat(),
|
||
}
|
||
except SQLAlchemyError as e:
|
||
logger.error(f"Ошибка базы данных при получении подписки для пользователя {telegram_id}: {e}")
|
||
raise HTTPException(status_code=500, detail="Database error")
|
||
except HTTPException as e:
|
||
# Пропускаем HTTPException, чтобы FastAPI обработал её автоматически
|
||
raise e
|
||
except Exception as e:
|
||
logger.error(f"Неожиданная ошибка: {e}")
|
||
raise HTTPException(status_code=500, detail="Internal Server Error")
|
||
|
||
@router.get("/subscriptions/{telegram_id}", response_model=List[SubscriptionResponse])
|
||
async def get_subscriptions(telegram_id: int, database_manager: DatabaseManager = Depends(get_database_manager)):
|
||
"""
|
||
Возвращает список подписок пользователя.
|
||
"""
|
||
logger.info(f"Получение подписок для пользователя: {telegram_id}")
|
||
try:
|
||
# Получаем подписки без ограничений или с указанным лимитом
|
||
subscriptions = await database_manager.get_last_subscriptions(telegram_id=telegram_id)
|
||
|
||
if not subscriptions:
|
||
logger.warning(f"Подписки для пользователя {telegram_id} не найдены")
|
||
raise HTTPException(status_code=404, detail="No subscriptions found")
|
||
|
||
# Формируем список подписок для ответа
|
||
return [
|
||
{
|
||
"id": sub.id,
|
||
"plan": sub.plan,
|
||
"vpn_server_id": sub.vpn_server_id,
|
||
"expiry_date": sub.expiry_date.isoformat(),
|
||
"created_at": sub.created_at.isoformat(),
|
||
"updated_at": sub.updated_at.isoformat(),
|
||
}
|
||
for sub in subscriptions
|
||
]
|
||
except SQLAlchemyError as e:
|
||
logger.error(f"Ошибка базы данных при получении подписок для пользователя {telegram_id}: {e}")
|
||
raise HTTPException(status_code=500, detail="Database error")
|
||
except Exception as e:
|
||
logger.error(f"Неожиданная ошибка: {e}")
|
||
raise HTTPException(status_code=500, detail=str(e))
|
||
|
||
|
||
@router.get("/uri", response_model=dict)
|
||
async def get_uri(telegram_id: int, database_manager: DatabaseManager = Depends(get_database_manager)):
|
||
"""
|
||
Возвращает список подписок пользователя.
|
||
"""
|
||
logger.info(f"Получение подписок для пользователя: {telegram_id}")
|
||
try:
|
||
# Получаем подписки без ограничений или с указанным лимитом
|
||
uri = await database_manager.generate_uri(telegram_id)
|
||
if uri == "SUB_ERROR":
|
||
raise HTTPException(status_code=404, detail="SUB_ERROR")
|
||
if not uri:
|
||
logger.warning(f"Не удалось сгенерировать URI для пользователя с telegram_id {telegram_id}")
|
||
raise HTTPException(status_code=404, detail="URI not found")
|
||
|
||
return {"detail": uri}
|
||
|
||
except HTTPException as e:
|
||
# Пропускаем HTTPException, чтобы FastAPI обработал её автоматически
|
||
raise e
|
||
|
||
except SQLAlchemyError as e:
|
||
logger.error(f"Ошибка базы данных при получении подписок для пользователя {telegram_id}: {e}")
|
||
raise HTTPException(status_code=500, detail="Database error")
|
||
|
||
except Exception as e:
|
||
logger.error(f"Неожиданная ошибка: {e}")
|
||
raise HTTPException(status_code=500, detail=str(e))
|