mirror of
https://github.com/BEDOLAGA-DEV/remnawave-bedolaga-telegram-bot.git
synced 2026-02-28 07:11:37 +00:00
- 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
142 lines
5.6 KiB
Python
142 lines
5.6 KiB
Python
from __future__ import annotations
|
|
|
|
from datetime import datetime
|
|
from typing import Annotated, Literal
|
|
|
|
from pydantic import BaseModel, Field, validator
|
|
|
|
|
|
CampaignBonusType = Annotated[
|
|
Literal['balance', 'subscription', 'none', 'tariff'],
|
|
Field(
|
|
description='Тип бонуса кампании: balance (баланс), subscription (пробная подписка), none (без награды), tariff (тариф)'
|
|
),
|
|
]
|
|
|
|
|
|
class CampaignBase(BaseModel):
|
|
name: str = Field(..., max_length=255)
|
|
start_parameter: str = Field(..., max_length=64, description='Start parameter для deep-link (уникальный)')
|
|
bonus_type: CampaignBonusType
|
|
balance_bonus_kopeks: int = Field(0, ge=0)
|
|
subscription_duration_days: int | None = Field(None, ge=0)
|
|
subscription_traffic_gb: int | None = Field(None, ge=0)
|
|
subscription_device_limit: int | None = Field(None, ge=0)
|
|
subscription_squads: list[str] = Field(default_factory=list)
|
|
# Поля для типа "tariff"
|
|
tariff_id: int | None = Field(None, ge=1, description='ID тарифа для выдачи')
|
|
tariff_duration_days: int | None = Field(None, ge=1, description='Длительность тарифа в днях')
|
|
|
|
@validator('name', 'start_parameter')
|
|
def strip_strings(cls, value: str) -> str:
|
|
return value.strip()
|
|
|
|
|
|
class CampaignCreateRequest(CampaignBase):
|
|
is_active: bool = True
|
|
|
|
@validator('balance_bonus_kopeks')
|
|
def validate_balance_bonus(cls, value: int, values: dict) -> int:
|
|
if values.get('bonus_type') == 'balance' and value <= 0:
|
|
raise ValueError('balance_bonus_kopeks must be positive for balance bonus')
|
|
return value
|
|
|
|
@validator('subscription_duration_days')
|
|
def validate_subscription_bonus(cls, value: int | None, values: dict):
|
|
if values.get('bonus_type') == 'subscription':
|
|
if value is None or value <= 0:
|
|
raise ValueError('subscription_duration_days must be positive for subscription bonus')
|
|
return value
|
|
|
|
@validator('tariff_id')
|
|
def validate_tariff_id(cls, value: int | None, values: dict):
|
|
if values.get('bonus_type') == 'tariff':
|
|
if value is None or value <= 0:
|
|
raise ValueError('tariff_id must be specified for tariff bonus')
|
|
return value
|
|
|
|
@validator('tariff_duration_days')
|
|
def validate_tariff_duration(cls, value: int | None, values: dict):
|
|
if values.get('bonus_type') == 'tariff':
|
|
if value is None or value <= 0:
|
|
raise ValueError('tariff_duration_days must be positive for tariff bonus')
|
|
return value
|
|
|
|
|
|
class CampaignResponse(BaseModel):
|
|
id: int
|
|
name: str
|
|
start_parameter: str
|
|
bonus_type: CampaignBonusType
|
|
balance_bonus_kopeks: int
|
|
balance_bonus_rubles: float
|
|
subscription_duration_days: int | None = None
|
|
subscription_traffic_gb: int | None = None
|
|
subscription_device_limit: int | None = None
|
|
subscription_squads: list[str] = Field(default_factory=list)
|
|
# Поля для типа "tariff"
|
|
tariff_id: int | None = None
|
|
tariff_duration_days: int | None = None
|
|
tariff_name: str | None = None # Для отображения названия тарифа
|
|
is_active: bool
|
|
created_by: int | None = None
|
|
created_at: datetime
|
|
updated_at: datetime
|
|
registrations_count: int = 0
|
|
|
|
|
|
class CampaignListResponse(BaseModel):
|
|
items: list[CampaignResponse]
|
|
total: int
|
|
limit: int
|
|
offset: int
|
|
|
|
|
|
class CampaignUpdateRequest(BaseModel):
|
|
name: str | None = Field(None, max_length=255)
|
|
start_parameter: str | None = Field(None, max_length=64)
|
|
bonus_type: CampaignBonusType | None = None
|
|
balance_bonus_kopeks: int | None = Field(None, ge=0)
|
|
subscription_duration_days: int | None = Field(None, ge=0)
|
|
subscription_traffic_gb: int | None = Field(None, ge=0)
|
|
subscription_device_limit: int | None = Field(None, ge=0)
|
|
subscription_squads: list[str] | None = None
|
|
# Поля для типа "tariff"
|
|
tariff_id: int | None = Field(None, ge=1)
|
|
tariff_duration_days: int | None = Field(None, ge=1)
|
|
is_active: bool | None = None
|
|
|
|
@validator('name', 'start_parameter', pre=True)
|
|
def strip_optional_strings(cls, value: str | None):
|
|
if isinstance(value, str):
|
|
return value.strip()
|
|
return value
|
|
|
|
@validator('balance_bonus_kopeks')
|
|
def validate_balance_bonus(cls, value: int | None, values: dict):
|
|
bonus_type = values.get('bonus_type')
|
|
if bonus_type == 'balance' and value is not None and value <= 0:
|
|
raise ValueError('balance_bonus_kopeks must be positive for balance bonus')
|
|
return value
|
|
|
|
@validator('subscription_duration_days')
|
|
def validate_subscription_bonus(cls, value: int | None, values: dict):
|
|
bonus_type = values.get('bonus_type')
|
|
if bonus_type == 'subscription' and value is not None and value <= 0:
|
|
raise ValueError('subscription_duration_days must be positive for subscription bonus')
|
|
return value
|
|
|
|
@validator('tariff_id')
|
|
def validate_tariff_id(cls, value: int | None, values: dict):
|
|
bonus_type = values.get('bonus_type')
|
|
if bonus_type == 'tariff' and value is not None and value <= 0:
|
|
raise ValueError('tariff_id must be positive for tariff bonus')
|
|
return value
|
|
|
|
@validator('tariff_duration_days')
|
|
def validate_tariff_duration(cls, value: int | None, values: dict):
|
|
bonus_type = values.get('bonus_type')
|
|
if bonus_type == 'tariff' and value is not None and value <= 0:
|
|
raise ValueError('tariff_duration_days must be positive for tariff bonus')
|
|
return value
|