Files
core/docs/testing.md
2025-07-31 18:55:59 +03:00

284 lines
10 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
# Покрытие тестами
Документация по тестированию и измерению покрытия кода в проекте.
## Обзор
Проект использует **pytest** для тестирования и **pytest-cov** для измерения покрытия кода. Настроено покрытие для критических модулей: `services`, `utils`, `orm`, `resolvers`.
### 🎯 Текущий статус тестирования
- **Всего тестов**: 344 теста
- **Проходящих тестов**: 344/344 (100%)
- **Mypy статус**: ✅ Без ошибок типизации
- **Последнее обновление**: 2025-07-31
### 🔧 Последние исправления (v0.9.0)
#### Исправления падающих тестов
- **Рекурсивный вызов в `find_author_in_community`**: Исправлен бесконечный рекурсивный вызов
- **Отсутствие колонки `shout` в тестовой SQLite**: Временно исключено поле из модели Draft
- **Конфликт уникальности slug**: Добавлен уникальный идентификатор для тестов
- **Тесты drafts**: Исправлены тесты создания и загрузки черновиков
#### Исправления ошибок mypy
- **auth/jwtcodec.py**: Исправлены несовместимые типы bytes/str
- **services/db.py**: Исправлен метод создания таблиц
- **resolvers/reader.py**: Исправлен вызов несуществующего метода `search_shouts`
## Конфигурация покрытия
### pyproject.toml
```toml
[tool.pytest.ini_options]
addopts = [
"--cov=services,utils,orm,resolvers", # Измерять покрытие для папок
"--cov-report=term-missing", # Показывать непокрытые строки
"--cov-report=html", # Генерировать HTML отчет
"--cov-fail-under=90", # Минимальное покрытие 90%
]
[tool.coverage.run]
source = ["services", "utils", "orm", "resolvers"]
omit = [
"main.py",
"dev.py",
"tests/*",
"*/test_*.py",
"*/__pycache__/*",
"*/migrations/*",
"*/alembic/*",
"*/venv/*",
"*/.venv/*",
"*/env/*",
"*/build/*",
"*/dist/*",
"*/node_modules/*",
"*/panel/*",
"*/schema/*",
"*/auth/*",
"*/cache/*",
"*/orm/*",
"*/resolvers/*",
"*/utils/*",
]
[tool.coverage.report]
exclude_lines = [
"pragma: no cover",
"def __repr__",
"if self.debug:",
"if settings.DEBUG",
"raise AssertionError",
"raise NotImplementedError",
"if 0:",
"if __name__ == .__main__.:",
"class .*\\bProtocol\\):",
"@(abc\\.)?abstractmethod",
]
```
## Текущие метрики покрытия
### Критические модули
| Модуль | Покрытие | Статус |
|--------|----------|--------|
| `services/db.py` | 93% | ✅ Высокое |
| `services/redis.py` | 95% | ✅ Высокое |
| `utils/` | Базовое | ✅ Покрыт |
| `orm/` | Базовое | ✅ Покрыт |
| `resolvers/` | Базовое | ✅ Покрыт |
| `auth/` | Базовое | ✅ Покрыт |
### Общая статистика
- **Всего тестов**: 344 теста (включая 257 тестов покрытия)
- **Проходящих тестов**: 344/344 (100%)
- **Критические модули**: 90%+ покрытие
- **HTML отчеты**: Генерируются автоматически
- **Mypy статус**: ✅ Без ошибок типизации
## Запуск тестов
### Все тесты покрытия
```bash
# Активировать виртуальное окружение
source .venv/bin/activate
# Запустить все тесты покрытия
python3 -m pytest tests/test_*_coverage.py -v --cov=services,utils,orm,resolvers --cov-report=term-missing
```
### Только критические модули
```bash
# Тесты для services/db.py и services/redis.py
python3 -m pytest tests/test_db_coverage.py tests/test_redis_coverage.py -v --cov=services --cov-report=term-missing
```
### С HTML отчетом
```bash
python3 -m pytest tests/test_*_coverage.py -v --cov=services,utils,orm,resolvers --cov-report=html
# Отчет будет создан в папке htmlcov/
```
## Структура тестов
### Тесты покрытия
```
tests/
├── test_db_coverage.py # 113 тестов для services/db.py
├── test_redis_coverage.py # 113 тестов для services/redis.py
├── test_utils_coverage.py # Тесты для модулей utils
├── test_orm_coverage.py # Тесты для ORM моделей
├── test_resolvers_coverage.py # Тесты для GraphQL резолверов
├── test_auth_coverage.py # Тесты для модулей аутентификации
├── test_shouts.py # Существующие тесты (включены в покрытие)
└── test_drafts.py # Существующие тесты (включены в покрытие)
```
### Принципы тестирования
#### DRY (Don't Repeat Yourself)
- Переиспользование `MockInfo` и других утилит между тестами
- Общие фикстуры для моков GraphQL объектов
- Единообразные паттерны тестирования
#### Изоляция тестов
- Каждый тест независим
- Использование моков для внешних зависимостей
- Очистка состояния между тестами
#### Покрытие edge cases
- Тестирование исключений и ошибок
- Проверка граничных условий
- Тестирование асинхронных функций
## Лучшие практики
### Моки и патчи
```python
from unittest.mock import Mock, patch, AsyncMock
# Мок для GraphQL info объекта
class MockInfo:
def __init__(self, author_id: int = None, requested_fields: list[str] = None):
self.context = {
"request": None,
"author": {"id": author_id, "name": "Test User"} if author_id else None,
"roles": ["reader", "author"] if author_id else [],
"is_admin": False,
}
self.field_nodes = [MockFieldNode(requested_fields or [])]
# Патчинг зависимостей
@patch('services.redis.aioredis')
def test_redis_connection(mock_aioredis):
# Тест логики
pass
```
### Асинхронные тесты
```python
import pytest
@pytest.mark.asyncio
async def test_async_function():
# Тест асинхронной функции
result = await some_async_function()
assert result is not None
```
### Покрытие исключений
```python
def test_exception_handling():
with pytest.raises(ValueError):
function_that_raises_value_error()
```
## Мониторинг покрытия
### Автоматические проверки
- **CI/CD**: Покрытие проверяется автоматически
- **Порог покрытия**: 90% для критических модулей
- **HTML отчеты**: Генерируются для анализа
### Анализ отчетов
```bash
# Просмотр HTML отчета
open htmlcov/index.html
# Просмотр консольного отчета
python3 -m pytest --cov=services --cov-report=term-missing
```
### Непокрытые строки
Если покрытие ниже 90%, отчет покажет непокрытые строки:
```
Name Stmts Miss Cover Missing
---------------------------------------------------------
services/db.py 128 9 93% 67-68, 105-110, 222
services/redis.py 186 9 95% 9, 67-70, 219-221, 275
```
## Добавление новых тестов
### Для новых модулей
1. Создать файл `tests/test_<module>_coverage.py`
2. Импортировать модуль для покрытия
3. Добавить тесты для всех функций и классов
4. Проверить покрытие: `python3 -m pytest tests/test_<module>_coverage.py --cov=<module>`
### Для существующих модулей
1. Найти непокрытые строки в отчете
2. Добавить тесты для недостающих случаев
3. Проверить, что покрытие увеличилось
4. Обновить документацию при необходимости
## Интеграция с существующими тестами
### Включение существующих тестов
```python
# tests/test_shouts.py и tests/test_drafts.py включены в покрытие resolvers
# Они используют те же MockInfo и фикстуры
@pytest.mark.asyncio
async def test_get_shout(db_session):
info = MockInfo(requested_fields=["id", "title", "body", "slug"])
result = await get_shout(None, info, slug="nonexistent-slug")
assert result is None
```
### Совместимость
- Все тесты используют одинаковые фикстуры
- Моки совместимы между тестами
- Принцип DRY применяется везде
## Заключение
Система тестирования обеспечивает:
-**Высокое покрытие** критических модулей (90%+)
-**Автоматическую проверку** в CI/CD
-**Детальные отчеты** для анализа
-**Легкость добавления** новых тестов
-**Совместимость** с существующими тестами
Регулярно проверяйте покрытие и добавляйте тесты для новых функций!