Display promo group discounts in mini app

This commit is contained in:
Egor
2025-10-09 03:44:54 +03:00
parent 67babf7566
commit acef77b0e8
3 changed files with 548 additions and 4 deletions

View File

@@ -9,8 +9,10 @@ from sqlalchemy.ext.asyncio import AsyncSession
from app.config import settings
from app.database.crud.server_squad import get_server_squad_by_uuid
from app.database.crud.promo_group import get_auto_assign_promo_groups
from app.database.crud.transaction import get_user_total_spent_kopeks
from app.database.crud.user import get_user_by_telegram_id
from app.database.models import Subscription, Transaction, User
from app.database.models import PromoGroup, Subscription, Transaction, User
from app.services.remnawave_service import (
RemnaWaveConfigurationError,
RemnaWaveService,
@@ -26,6 +28,7 @@ from ..dependencies import get_db_session
from ..schemas.miniapp import (
MiniAppConnectedServer,
MiniAppDevice,
MiniAppAutoPromoGroupLevel,
MiniAppPromoGroup,
MiniAppSubscriptionRequest,
MiniAppSubscriptionResponse,
@@ -336,6 +339,27 @@ async def get_subscription_details(
balance_currency = balance_currency.upper()
promo_group = getattr(user, "promo_group", None)
total_spent_kopeks = await get_user_total_spent_kopeks(db, user.id)
auto_assign_groups = await get_auto_assign_promo_groups(db)
auto_promo_levels: List[MiniAppAutoPromoGroupLevel] = []
for group in auto_assign_groups:
threshold = group.auto_assign_total_spent_kopeks or 0
if threshold <= 0:
continue
auto_promo_levels.append(
MiniAppAutoPromoGroupLevel(
id=group.id,
name=group.name,
threshold_kopeks=threshold,
threshold_rubles=round(threshold / 100, 2),
threshold_label=settings.format_price(threshold),
is_reached=total_spent_kopeks >= threshold,
is_current=bool(promo_group and promo_group.id == group.id),
**_extract_promo_discounts(group),
)
)
response_user = MiniAppSubscriptionUser(
telegram_id=user.telegram_id,
@@ -386,11 +410,66 @@ async def get_subscription_details(
balance_rubles=round(user.balance_rubles, 2),
balance_currency=balance_currency,
transactions=[_serialize_transaction(tx) for tx in transactions],
promo_group=MiniAppPromoGroup(id=promo_group.id, name=promo_group.name)
if promo_group
else None,
promo_group=(
MiniAppPromoGroup(
id=promo_group.id,
name=promo_group.name,
**_extract_promo_discounts(promo_group),
)
if promo_group
else None
),
auto_assign_promo_groups=auto_promo_levels,
total_spent_kopeks=total_spent_kopeks,
total_spent_rubles=round(total_spent_kopeks / 100, 2),
total_spent_label=settings.format_price(total_spent_kopeks),
subscription_type="trial" if subscription.is_trial else "paid",
autopay_enabled=bool(subscription.autopay_enabled),
branding=settings.get_miniapp_branding(),
)
def _safe_int(value: Any) -> int:
try:
return int(value)
except (TypeError, ValueError):
return 0
def _normalize_period_discounts(
raw: Optional[Dict[Any, Any]]
) -> Dict[int, int]:
if not isinstance(raw, dict):
return {}
normalized: Dict[int, int] = {}
for key, value in raw.items():
try:
period = int(key)
normalized[period] = int(value)
except (TypeError, ValueError):
continue
return normalized
def _extract_promo_discounts(group: Optional[PromoGroup]) -> Dict[str, Any]:
if not group:
return {
"server_discount_percent": 0,
"traffic_discount_percent": 0,
"device_discount_percent": 0,
"period_discounts": {},
"apply_discounts_to_addons": True,
}
return {
"server_discount_percent": max(0, _safe_int(getattr(group, "server_discount_percent", 0))),
"traffic_discount_percent": max(0, _safe_int(getattr(group, "traffic_discount_percent", 0))),
"device_discount_percent": max(0, _safe_int(getattr(group, "device_discount_percent", 0))),
"period_discounts": _normalize_period_discounts(getattr(group, "period_discounts", None)),
"apply_discounts_to_addons": bool(
getattr(group, "apply_discounts_to_addons", True)
),
}

View File

@@ -39,6 +39,26 @@ class MiniAppSubscriptionUser(BaseModel):
class MiniAppPromoGroup(BaseModel):
id: int
name: str
server_discount_percent: int = 0
traffic_discount_percent: int = 0
device_discount_percent: int = 0
period_discounts: Dict[int, int] = Field(default_factory=dict)
apply_discounts_to_addons: bool = True
class MiniAppAutoPromoGroupLevel(BaseModel):
id: int
name: str
threshold_kopeks: int
threshold_rubles: float
threshold_label: str
is_reached: bool = False
is_current: bool = False
server_discount_percent: int = 0
traffic_discount_percent: int = 0
device_discount_percent: int = 0
period_discounts: Dict[int, int] = Field(default_factory=dict)
apply_discounts_to_addons: bool = True
class MiniAppConnectedServer(BaseModel):
@@ -90,6 +110,10 @@ class MiniAppSubscriptionResponse(BaseModel):
balance_currency: Optional[str] = None
transactions: List[MiniAppTransaction] = Field(default_factory=list)
promo_group: Optional[MiniAppPromoGroup] = None
auto_assign_promo_groups: List[MiniAppAutoPromoGroupLevel] = Field(default_factory=list)
total_spent_kopeks: int = 0
total_spent_rubles: float = 0.0
total_spent_label: Optional[str] = None
subscription_type: str
autopay_enabled: bool = False
branding: Optional[MiniAppBranding] = None

View File

@@ -552,6 +552,10 @@
border-bottom: none;
}
.info-item.promo-group-info {
align-items: flex-start;
}
.info-item:hover {
padding-left: 8px;
background: rgba(var(--primary-rgb), 0.02);
@@ -576,6 +580,173 @@
text-align: right;
}
.info-item.promo-group-info .info-value {
display: flex;
flex-direction: column;
align-items: flex-end;
gap: 6px;
width: 100%;
}
.promo-group-discounts {
display: flex;
flex-wrap: wrap;
gap: 6px;
justify-content: flex-end;
}
/* Promo levels */
.promo-levels-card .card-content {
opacity: 1;
max-height: 2000px;
padding-bottom: 16px;
}
.promo-level-summary {
display: flex;
justify-content: space-between;
align-items: center;
padding: 0 20px 12px;
font-weight: 600;
color: var(--text-primary);
}
.promo-level-summary-label {
color: var(--text-secondary);
font-size: 13px;
}
.promo-level-summary-value {
font-size: 16px;
}
.promo-level-list {
list-style: none;
padding: 0 20px 8px;
margin: 0;
display: flex;
flex-direction: column;
gap: 12px;
}
.promo-level-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 14px 16px;
border-radius: var(--radius);
border: 1px solid var(--border-color);
background: var(--bg-secondary);
transition: border-color 0.3s ease, box-shadow 0.3s ease, transform 0.3s ease;
}
.promo-level-item:hover {
border-color: rgba(var(--primary-rgb), 0.4);
box-shadow: var(--shadow-sm);
transform: translateY(-1px);
}
.promo-level-item.current {
border-color: var(--primary);
box-shadow: var(--shadow-sm);
}
.promo-level-info {
display: flex;
flex-direction: column;
gap: 4px;
}
.promo-level-discounts {
display: flex;
flex-wrap: wrap;
gap: 6px;
margin-top: 6px;
}
.promo-level-name {
font-weight: 700;
color: var(--text-primary);
font-size: 15px;
}
.promo-level-threshold {
font-size: 13px;
color: var(--text-secondary);
}
.promo-level-badge {
padding: 6px 12px;
border-radius: 999px;
font-size: 12px;
font-weight: 600;
letter-spacing: 0.3px;
}
.promo-level-item.current .promo-level-badge {
background: rgba(var(--primary-rgb), 0.18);
color: var(--primary);
}
.promo-level-item.reached .promo-level-badge {
background: rgba(16, 185, 129, 0.18);
color: var(--success);
}
.promo-level-item.locked .promo-level-badge {
background: rgba(148, 163, 184, 0.18);
color: var(--text-secondary);
}
.promo-discount-badge {
display: inline-flex;
align-items: center;
gap: 4px;
padding: 4px 8px;
border-radius: 999px;
border: 1px solid rgba(var(--primary-rgb), 0.12);
background: rgba(var(--primary-rgb), 0.08);
color: var(--text-secondary);
font-size: 12px;
font-weight: 600;
white-space: nowrap;
}
.promo-discount-badge .promo-discount-value {
color: var(--text-primary);
}
.promo-discount-badge.muted {
background: rgba(var(--primary-rgb), 0.05);
border-color: rgba(var(--primary-rgb), 0.08);
}
.promo-discount-badge.muted .promo-discount-value {
color: var(--text-secondary);
}
.promo-level-item.current .promo-discount-badge {
border-color: rgba(var(--primary-rgb), 0.35);
color: var(--primary);
}
.promo-level-item.current .promo-discount-badge .promo-discount-value {
color: var(--primary);
}
.promo-level-item.reached .promo-discount-badge {
border-color: rgba(16, 185, 129, 0.35);
color: var(--success);
}
.promo-level-item.reached .promo-discount-badge .promo-discount-value {
color: var(--success);
}
.promo-level-item.locked .promo-discount-badge {
opacity: 0.8;
}
/* Balance Card */
.balance-card {
background: linear-gradient(135deg, rgba(var(--primary-rgb), 0.1), rgba(var(--primary-rgb), 0.05));
@@ -1242,6 +1413,21 @@
color: var(--text-secondary);
}
:root[data-theme="dark"] .promo-discount-badge {
background: rgba(var(--primary-rgb), 0.12);
border-color: rgba(var(--primary-rgb), 0.2);
}
:root[data-theme="dark"] .promo-discount-badge.muted {
background: rgba(148, 163, 184, 0.15);
border-color: rgba(148, 163, 184, 0.25);
color: rgba(226, 232, 240, 0.75);
}
:root[data-theme="dark"] .promo-discount-badge.muted .promo-discount-value {
color: rgba(226, 232, 240, 0.75);
}
:root[data-theme="dark"] body,
:root[data-theme="dark"] .card,
:root[data-theme="dark"] .user-card,
@@ -1380,6 +1566,13 @@
<span class="info-label" data-i18n="info.subscription_type">Type</span>
<span class="info-value" id="subscriptionType">-</span>
</div>
<div class="info-item promo-group-info">
<span class="info-label" data-i18n="info.promo_group">Promo group</span>
<div class="info-value">
<span id="promoGroupValue">-</span>
<div class="promo-group-discounts hidden" id="promoGroupDiscounts"></div>
</div>
</div>
<div class="info-item">
<span class="info-label" data-i18n="info.device_limit">Device Limit</span>
<span class="info-value" id="deviceLimit">-</span>
@@ -1391,6 +1584,28 @@
</div>
</div>
<!-- Promo Levels Card -->
<div class="card promo-levels-card" id="promoLevelsCard">
<div class="card-header">
<div class="card-title">
<svg class="card-icon" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M12 8c1.657 0 3-1.343 3-3s-1.343-3-3-3-3 1.343-3 3 1.343 3 3 3zm0 0v13m-4-5l4 2 4-2"/>
</svg>
<span data-i18n="card.promo_levels.title">Promo Levels</span>
</div>
</div>
<div class="card-content">
<div class="promo-level-summary">
<span class="promo-level-summary-label" data-i18n="promo_levels.total_spent">Total spent</span>
<span class="promo-level-summary-value" id="promoLevelsSpent"></span>
</div>
<ul class="promo-level-list" id="promoLevelsList"></ul>
<div class="empty-state hidden" id="promoLevelsEmpty" data-i18n="promo_levels.empty">
Automatic promo levels are not configured yet
</div>
</div>
</div>
<!-- Action Buttons -->
<div class="btn-group">
<button class="btn btn-primary" id="connectBtn">
@@ -1671,6 +1886,7 @@
'card.history.title': 'Transaction History',
'card.servers.title': 'Connected Servers',
'card.devices.title': 'Connected Devices',
'card.promo_levels.title': 'Promo Levels',
'apps.title': 'Installation guide',
'apps.no_data': 'No installation guide available for this platform yet.',
'apps.featured': 'Recommended',
@@ -1687,6 +1903,16 @@
'history.type.referral_reward': 'Referral reward',
'servers.empty': 'No servers connected yet',
'devices.empty': 'No devices connected yet',
'promo_levels.total_spent': 'Total spent',
'promo_levels.threshold': 'from {amount}',
'promo_levels.badge.current': 'Current level',
'promo_levels.badge.unlocked': 'Unlocked',
'promo_levels.badge.locked': 'Locked',
'promo_levels.discounts.server': 'Servers',
'promo_levels.discounts.traffic': 'Traffic',
'promo_levels.discounts.devices': 'Devices',
'promo_levels.discounts.none': 'No discounts',
'promo_levels.empty': 'Automatic promo levels are not configured yet',
'language.ariaLabel': 'Select interface language',
'notifications.copy.success': 'Subscription link copied to clipboard.',
'notifications.copy.failure': 'Unable to copy the subscription link automatically. Please copy it manually.',
@@ -1735,6 +1961,7 @@
'card.history.title': 'История операций',
'card.servers.title': 'Подключённые серверы',
'card.devices.title': 'Подключенные устройства',
'card.promo_levels.title': 'Уровни промогрупп',
'apps.title': 'Инструкция по установке',
'apps.no_data': 'Для этой платформы инструкция пока недоступна.',
'apps.featured': 'Рекомендуем',
@@ -1751,6 +1978,16 @@
'history.type.referral_reward': 'Реферальное вознаграждение',
'servers.empty': 'Подключённых серверов пока нет',
'devices.empty': 'Подключённых устройств пока нет',
'promo_levels.total_spent': 'Всего потрачено',
'promo_levels.threshold': 'от {amount}',
'promo_levels.badge.current': 'Текущий уровень',
'promo_levels.badge.unlocked': 'Получен',
'promo_levels.badge.locked': 'Недоступен',
'promo_levels.discounts.server': 'Серверы',
'promo_levels.discounts.traffic': 'Трафик',
'promo_levels.discounts.devices': 'Устройства',
'promo_levels.discounts.none': 'Скидок нет',
'promo_levels.empty': 'Автовыдача промогрупп ещё не настроена',
'language.ariaLabel': 'Выберите язык интерфейса',
'notifications.copy.success': 'Ссылка подписки скопирована.',
'notifications.copy.failure': 'Не удалось автоматически скопировать ссылку. Пожалуйста, сделайте это вручную.',
@@ -2289,6 +2526,8 @@
: autopayLabel;
}
renderPromoGroupInfo();
renderPromoLevels();
renderBalanceSection();
renderTransactionHistory();
renderServersList();
@@ -2495,6 +2734,21 @@
}
}
function formatPriceFromKopeks(kopeks, currency) {
const normalized = typeof kopeks === 'number'
? kopeks
: Number.parseInt(String(kopeks ?? '').trim() || '0', 10);
const currencyCode = currency
? String(currency).toUpperCase()
: String(userData?.balance_currency || 'RUB').toUpperCase();
if (!Number.isFinite(normalized)) {
return formatCurrency(0, currencyCode);
}
return formatCurrency(normalized / 100, currencyCode);
}
function formatDate(value) {
if (!value) {
return '—';
@@ -2695,6 +2949,193 @@
}).join('');
}
const PROMO_DISCOUNT_FIELDS = [
{ field: 'server_discount_percent', labelKey: 'promo_levels.discounts.server' },
{ field: 'traffic_discount_percent', labelKey: 'promo_levels.discounts.traffic' },
{ field: 'device_discount_percent', labelKey: 'promo_levels.discounts.devices' },
];
function normalizePromoPercent(value) {
if (typeof value === 'number' && Number.isFinite(value)) {
return value;
}
if (typeof value === 'string' && value.trim() !== '') {
const parsed = Number.parseInt(value, 10);
if (Number.isFinite(parsed)) {
return parsed;
}
}
return null;
}
function populatePromoDiscounts(container, source, options = {}) {
const { showEmptyMessage = false } = options;
if (!container) {
return false;
}
container.innerHTML = '';
if (!source) {
container.classList.add('hidden');
return false;
}
let hasNumeric = false;
PROMO_DISCOUNT_FIELDS.forEach(({ field, labelKey }) => {
const percentValue = normalizePromoPercent(source?.[field]);
if (percentValue === null) {
return;
}
hasNumeric = true;
const normalized = Math.max(0, Math.round(percentValue));
const badge = document.createElement('span');
badge.className = 'promo-discount-badge';
if (normalized === 0) {
badge.classList.add('muted');
}
const label = document.createElement('span');
label.className = 'promo-discount-label';
label.textContent = t(labelKey);
const valueElement = document.createElement('span');
valueElement.className = 'promo-discount-value';
valueElement.textContent = `${normalized}%`;
badge.appendChild(label);
badge.appendChild(valueElement);
container.appendChild(badge);
});
if (!hasNumeric) {
if (showEmptyMessage) {
const badge = document.createElement('span');
badge.className = 'promo-discount-badge muted';
badge.textContent = t('promo_levels.discounts.none');
container.appendChild(badge);
} else {
container.classList.add('hidden');
return false;
}
}
container.classList.remove('hidden');
return true;
}
function renderPromoGroupInfo() {
const valueElement = document.getElementById('promoGroupValue');
const discountsContainer = document.getElementById('promoGroupDiscounts');
if (!valueElement) {
return;
}
const promoGroup = userData?.promo_group;
const promoGroupName = promoGroup?.name;
valueElement.textContent = promoGroupName || t('values.not_available');
if (discountsContainer) {
const hasDiscounts = populatePromoDiscounts(discountsContainer, promoGroup, {
showEmptyMessage: true,
});
discountsContainer.classList.toggle('hidden', !hasDiscounts);
}
}
function renderPromoLevels() {
const list = document.getElementById('promoLevelsList');
const emptyState = document.getElementById('promoLevelsEmpty');
const totalSpentElement = document.getElementById('promoLevelsSpent');
const card = document.getElementById('promoLevelsCard');
if (!list || !emptyState || !totalSpentElement || !card) {
return;
}
const levels = Array.isArray(userData?.auto_assign_promo_groups)
? userData.auto_assign_promo_groups
: [];
const totalSpentKopeksRaw = typeof userData?.total_spent_kopeks === 'number'
? userData.total_spent_kopeks
: Number.parseInt(userData?.total_spent_kopeks ?? '0', 10);
const totalSpentKopeks = Number.isFinite(totalSpentKopeksRaw) ? totalSpentKopeksRaw : 0;
totalSpentElement.textContent = userData?.total_spent_label
|| formatPriceFromKopeks(totalSpentKopeks);
list.innerHTML = '';
if (!levels.length) {
card.classList.add('hidden');
emptyState.classList.add('hidden');
return;
}
card.classList.remove('hidden');
emptyState.classList.add('hidden');
const currencyCode = (userData?.balance_currency || 'RUB').toUpperCase();
const thresholdTemplate = t('promo_levels.threshold');
levels.forEach(level => {
const classes = ['promo-level-item'];
if (level?.is_current) {
classes.push('current', 'reached');
} else if (level?.is_reached) {
classes.push('reached');
} else {
classes.push('locked');
}
const item = document.createElement('li');
item.className = classes.join(' ');
const info = document.createElement('div');
info.className = 'promo-level-info';
const name = document.createElement('div');
name.className = 'promo-level-name';
name.textContent = level?.name || t('values.not_available');
info.appendChild(name);
const threshold = document.createElement('div');
threshold.className = 'promo-level-threshold';
const thresholdLabel = level?.threshold_label
|| formatPriceFromKopeks(level?.threshold_kopeks, currencyCode);
threshold.textContent = thresholdTemplate.includes('{amount}')
? thresholdTemplate.replace('{amount}', thresholdLabel)
: `${thresholdTemplate} ${thresholdLabel}`;
info.appendChild(threshold);
const discounts = document.createElement('div');
discounts.className = 'promo-level-discounts';
if (populatePromoDiscounts(discounts, level)) {
info.appendChild(discounts);
}
const badge = document.createElement('div');
badge.className = 'promo-level-badge';
let badgeKey = 'promo_levels.badge.locked';
if (level?.is_current) {
badgeKey = 'promo_levels.badge.current';
} else if (level?.is_reached) {
badgeKey = 'promo_levels.badge.unlocked';
}
badge.textContent = t(badgeKey);
item.appendChild(info);
item.appendChild(badge);
list.appendChild(item);
});
}
function getCurrentSubscriptionUrl() {
return userData?.subscription_url || userData?.subscriptionUrl || '';
}