"""Тесты для базовых форматтеров из app.utils.formatters.""" from datetime import datetime, timedelta from app.utils import formatters def test_format_datetime_handles_iso_strings(fixed_datetime: datetime) -> None: """ISO-строка должна корректно преобразовываться в отформатированный текст.""" iso_value = fixed_datetime.isoformat() assert formatters.format_datetime(iso_value) == fixed_datetime.strftime('%d.%m.%Y %H:%M') def test_format_date_uses_custom_format(fixed_datetime: datetime) -> None: """Можно задавать собственный шаблон вывода.""" iso_value = fixed_datetime.isoformat() assert formatters.format_date(iso_value, format_str='%Y/%m/%d') == fixed_datetime.strftime('%Y/%m/%d') def test_format_time_ago_returns_human_readable_text() -> None: """Разница во времени должна переводиться в человеко-понятную строку.""" point_in_time = datetime.utcnow() - timedelta(minutes=5) assert formatters.format_time_ago(point_in_time, language='ru') == '5 мин. назад' assert formatters.format_time_ago(point_in_time, language='en') == '5 minutes ago' def test_format_days_declension_handles_russian_rules() -> None: """Склонение дней в русском языке зависит от числа.""" assert formatters.format_days_declension(1) == '1 день' assert formatters.format_days_declension(3) == '3 дня' assert formatters.format_days_declension(10) == '10 дней' def test_format_duration_switches_units() -> None: """В зависимости от длины интервала выбирается подходящая единица измерения.""" assert formatters.format_duration(45) == '45 сек.' assert formatters.format_duration(120) == '2 мин.' assert formatters.format_duration(7200) == '2 ч.' assert formatters.format_duration(172800) == '2 дн.' def test_format_bytes_scales_value() -> None: """Размер должен выражаться в наиболее подходящей единице.""" assert formatters.format_bytes(0) == '0 B' assert formatters.format_bytes(1024) == '1 KB' assert formatters.format_bytes(1024 * 1024) == '1 MB' def test_format_percentage_respects_precision() -> None: """Проценты форматируются с нужным количеством знаков.""" assert formatters.format_percentage(12.3456, decimals=2) == '12.35%' def test_format_number_inserts_separators() -> None: """Разделители тысяч должны расставляться корректно как для int, так и для float.""" assert formatters.format_number(1234567) == '1 234 567' assert formatters.format_number(1234.56) == '1 234.55' def test_truncate_text_appends_suffix() -> None: """Строки, превышающие лимит, должны обрезаться и дополняться суффиксом.""" source = 'a' * 10 assert formatters.truncate_text(source, max_length=5) == 'aa...' def test_format_username_prefers_full_name() -> None: """Полное имя имеет приоритет, затем username, затем ID.""" assert formatters.format_username('nickname', 1, full_name='Имя') == 'Имя' assert formatters.format_username('nickname', 1, full_name=None) == '@nickname' assert formatters.format_username(None, 42, full_name=None) == 'ID42' def test_format_subscription_status_handles_active_and_expired() -> None: """Статус подписки различается для активных/просроченных случаев.""" future = datetime.utcnow() + timedelta(days=2) active = formatters.format_subscription_status( is_active=True, is_trial=False, end_date=future, language='ru', ) assert active.startswith('✅ Активна') assert '(' in active and ')' in active past = datetime.utcnow() - timedelta(days=1) expired = formatters.format_subscription_status( is_active=True, is_trial=False, end_date=past, language='ru', ) assert expired == '⏰ Истекла' def test_format_traffic_usage_supports_unlimited() -> None: """При безлимитном тарифе в строке должна появляться бесконечность.""" assert formatters.format_traffic_usage(50.0, 0, language='ru') == '50.0 ГБ / ∞' assert formatters.format_traffic_usage(10.0, 100, language='ru') == '10.0 ГБ / 100 ГБ (10.0%)' def test_format_boolean_localises_output() -> None: """Булевые значения отображаются локализованными словами.""" assert formatters.format_boolean(True, language='ru') == '✅ Да' assert formatters.format_boolean(False, language='en') == '❌ No'