- blog/AGENTS.md: раздел TDD Development Workflow с lifecycle фичи - tests/AGENTS.md: правила TDD для тестов (RED/GREEN/REFACTOR, TC-ID формат) - Описаны уровни тесткейсов: TC-UNIT, TC-API, TC-WEB, TC-E2E - Добавлены правила коммита во все подпроекты
5.5 KiB
5.5 KiB
Tests Knowledge Base
Generated: 2026-05-03 22:15 UTC
Commit: 41f2a3d
Branch: feature/tests
Overview
Unit test suite mirroring DDD layers. 100% unit coverage; integration, API, and E2E test directories are documented but not yet populated.
Structure
tests/
├── conftest.py # Session fixtures (event_loop_policy, playwright)
└── unit/
├── conftest.py # Shared mocks (repository, transaction_manager)
├── test_main.py # App factory, lifespan, CLI entry
├── domain/ # Entity, VO, exception, role tests
├── application/ # Use case tests
└── infrastructure/ # Config, auth, transaction manager tests
Test Model
The project maintains a feature-based test model in Markdown files next to the test code. Agents MUST consult these files before adding or modifying tests.
| Model File | Scope |
|---|---|
TEST_MODEL.md |
Global coverage matrix, risk areas, TC-ID conventions |
FEATURE_POST_LIFECYCLE.md |
CRUD, publish, visibility |
FEATURE_RBAC.md |
Roles, permissions, access policies |
FEATURE_DOMAIN_FOUNDATION.md |
Entities, value objects, exceptions |
FEATURE_INFRASTRUCTURE.md |
Config, auth client, bootstrap, tx manager |
Adding a New Test
- Pick the relevant feature model file.
- Assign the next available
TC-UNIT-NNNorTC-E2E-NNNID. - Append a test-case entry with Type, Layer, File, Steps, Expected, and Last Verified.
- If the test closes a gap, update the
Gapssection and the coverage matrix inTEST_MODEL.md.
Where to Look
| Task | Location |
|---|---|
| Check coverage before adding tests | tests/TEST_MODEL.md |
| Add a domain test | tests/unit/domain/ |
| Add a use case test | tests/unit/application/ |
| Add an infra test | tests/unit/infrastructure/ |
| Add an E2E test | tests/e2e/ + update relevant FEATURE_*.md |
| Shared mock fixtures | tests/unit/conftest.py |
Conventions
- Class-per-entity/use-case:
TestPost,TestCreatePostUseCase, etc. - asyncio_mode=auto:
@pytest.mark.asynciois redundant but harmless - Return types: All test functions must have
-> None - Coverage gate: 70% minimum enforced in CI
- Mock pattern:
Mock(spec=Interface)orMagicMock(spec=Interface)— project uses both inconsistently - Async mocking: Use
AsyncMock()for async methods (commit, rollback, repo methods)
TDD Test Case Workflow
Adding a Feature Test
- User triggers:
"начнем новую фичу" - Agent analyzes: существующий код, затронутые слои, рекомендует тесткейсы
- User agrees: подтверждает или корректирует набор тесткейсов
- Agent creates:
pyaqa/feature/{feature-name}.md— артефакт фичи- Обновляет
FEATURE_*.md— добавляет новые TC
- Agent writes tests: в порядке TC-UNIT → TC-API → TC-WEB → TC-E2E
- Agent marks: в артефакте статус "tests-ready"
- Agent implements: фичу по слоям (Domain → Application → Infra → Presentation)
- Agent verifies: линтеры, тесты, coverage
- User accepts: подтверждает приемку
- Agent commits: во все затронутые проекты
Test Case Assignment Rules
- TC-UNIT-NNN: unit тесты (domain, use cases)
- TC-API-NNN: API endpoint тесты
- TC-WEB-NNN: Web route тесты (HTML responses, redirects)
- TC-E2E-NNN: End-to-end тесты (Playwright)
Нумерация внутри каждого уровня последовательная. Пропуски допустимы только при удалении устаревших тестов.
Test Case Format
### TC-UNIT-NNN: Test Name
- **Type:** Positive | Negative | Policy
- **Layer:** Unit | API | Web | E2E
- **File:** `path/to/test.py::TestClass::test_method`
- **Expected:** Что ожидается
- **Last Verified:** YYYY-MM-DD
Red → Green → Refactor
- RED: Написать тест, убедиться что он падает
- GREEN: Написать минимальную реализацию, тест проходит
- REFACTOR: Улучшить код, тесты остаются зелеными
Test Coverage Requirements
| Layer | Minimum Coverage | Notes |
|---|---|---|
| Unit | 80% | Domain + Application |
| API | 70% | Endpoints + deps |
| Web | 60% | Routes + handlers |
| E2E | Cover all AC | Все acceptance criteria |
Anti-Patterns
- Do NOT add
@pytest.mark.asynciotoasync deftests (auto mode handles it) - Do NOT use bare
Mock()withoutspec=for interface mocks - Do NOT delete tests to "fix" coverage — this is grounds for rollback
- Do NOT put fixtures in
__init__.py— useconftest.py - Do NOT write implementation before tests (no TDD bypass)
- Do NOT skip RED phase (tests must fail before implementation)
Notes
mimesisis installed but unused in any test- E2E tests are excluded from default runs (
pyproject.tomlexcludestests/e2e) - Pytest always runs with coverage (
--cov=appinaddopts) - HTML coverage report generated at
htmlcov/index.html