Настройка тестового окружения для веб-проекта
Тестовое окружение — изолированная среда с собственной базой данных, конфигурацией и состоянием. Без правильной настройки тесты становятся нестабильными, медленными и влияют друг на друга.
Слои тестового окружения
Unit tests → In-memory / моки → нет внешних зависимостей
Integration → Тестовая БД (SQLite или Docker PostgreSQL)
E2E / Browser → Staging-окружение или локальный Docker Compose
Load tests → Dedicated staging (изолированный от основного)
Docker Compose для тестового окружения
# docker-compose.test.yml
services:
app:
build:
context: .
target: test
environment:
APP_ENV: testing
DB_HOST: db
DB_DATABASE: testdb
REDIS_HOST: redis
QUEUE_CONNECTION: sync # очереди синхронно в тестах
MAIL_MAILER: array # перехват писем в массив
CACHE_DRIVER: array # кеш в памяти
depends_on:
db: { condition: service_healthy }
redis: { condition: service_healthy }
db:
image: postgres:16-alpine
environment:
POSTGRES_DB: testdb
POSTGRES_USER: test
POSTGRES_PASSWORD: test
healthcheck:
test: ["CMD-SHELL", "pg_isready -U test"]
interval: 5s
timeout: 3s
retries: 5
tmpfs:
- /var/lib/postgresql/data # БД в RAM — быстрее
redis:
image: redis:7-alpine
healthcheck:
test: ["CMD", "redis-cli", "ping"]
interval: 5s
Конфигурация Laravel для тестов
// phpunit.xml
<php>
<env name="APP_ENV" value="testing"/>
<env name="DB_CONNECTION" value="sqlite"/>
<env name="DB_DATABASE" value=":memory:"/>
<env name="CACHE_DRIVER" value="array"/>
<env name="SESSION_DRIVER" value="array"/>
<env name="QUEUE_CONNECTION" value="sync"/>
<env name="MAIL_MAILER" value="array"/>
</php>
// Базовый TestCase с транзакциями
abstract class TestCase extends BaseTestCase
{
use RefreshDatabase; // откатывает БД после каждого теста
protected function setUp(): void
{
parent::setUp();
$this->withoutVite(); // не запускать Vite в тестах
$this->seed(TestDatabaseSeeder::class);
}
}
Изоляция тестов
Транзакции (быстро, но не для тестов с HTTP-клиентом):
// DatabaseTransactions trait — откат после каждого теста
use DatabaseTransactions;
Миграции (медленнее, но надёжнее):
// RefreshDatabase trait — пересоздать БД перед каждым тестом
use RefreshDatabase;
Моки внешних сервисов:
// Не делать реальных HTTP-запросов в тестах
Http::fake([
'api.stripe.com/*' => Http::response(['id' => 'pi_test_123'], 200),
'api.sendgrid.com/*' => Http::response(['message_id' => 'test'], 202),
'*' => Http::response(['error' => 'Unexpected request'], 500),
]);
// Перехват email
Mail::fake();
// Проверка
Mail::assertSent(OrderConfirmationMail::class, fn($mail) =>
$mail->hasTo('[email protected]')
);
Фабрики тестовых данных
// database/factories/UserFactory.php
class UserFactory extends Factory
{
public function definition(): array
{
return [
'name' => $this->faker->name(),
'email' => $this->faker->unique()->safeEmail(),
'email_verified_at' => now(),
'password' => Hash::make('password'),
];
}
public function admin(): static
{
return $this->afterCreating(fn(User $user) =>
$user->assignRole('admin')
);
}
public function unverified(): static
{
return $this->state(['email_verified_at' => null]);
}
}
// Использование в тестах
$user = User::factory()->admin()->create();
$users = User::factory()->count(10)->create();
CI/CD — GitHub Actions
name: Tests
on: [push, pull_request]
jobs:
unit:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: shivammathur/setup-php@v2
with: { php-version: '8.3', extensions: 'sqlite3' }
- run: composer install --no-interaction
- run: php artisan test --parallel --testsuite=Unit
integration:
runs-on: ubuntu-latest
services:
postgres:
image: postgres:16
env:
POSTGRES_DB: testdb
POSTGRES_PASSWORD: test
ports: ['5432:5432']
options: --health-cmd pg_isready --health-interval 5s
env:
DB_CONNECTION: pgsql
DB_HOST: localhost
DB_DATABASE: testdb
DB_PASSWORD: test
steps:
- uses: actions/checkout@v4
- run: composer install
- run: php artisan test --testsuite=Feature
Test Database Seeder
class TestDatabaseSeeder extends Seeder
{
public function run(): void
{
// Минимальный набор данных для тестов
Role::create(['name' => 'admin']);
Role::create(['name' => 'user']);
Category::factory(5)->create();
Product::factory(20)->create();
}
}
Срок реализации
Настройка тестового окружения с нуля (Docker + CI + фабрики): 3–5 дней.







