본문으로 건너뛰기
버전: 9.x

데이터베이스 테스트 (Database Testing)

소개

라라벨은 데이터베이스 기반 애플리케이션의 테스트를 더 쉽게 만들어주는 다양한 유용한 도구와 assertion(어설션, 검증 메서드)을 제공합니다. 또한 라라벨의 모델 팩토리와 시더 기능을 활용하면, 여러분의 애플리케이션에서 Eloquent 모델과 연관관계를 사용하여 테스트용 데이터베이스 레코드를 쉽게 만들 수 있습니다. 이 문서에서는 이러한 강력한 기능들을 모두 다룹니다.

각 테스트 후 데이터베이스 초기화

자세한 내용으로 들어가기 전에, 각 테스트가 끝날 때마다 데이터베이스를 초기화하여 이전 테스트의 데이터가 이후 테스트에 영향을 주지 않도록 처리하는 방법을 먼저 설명합니다. 라라벨에 포함된 Illuminate\Foundation\Testing\RefreshDatabase 트레이트를 사용하면 이 과정을 간단하게 처리할 수 있습니다. 테스트 클래스에서 아래와 같이 해당 트레이트를 사용하세요:

<?php

namespace Tests\Feature;

use Illuminate\Foundation\Testing\RefreshDatabase;
use Illuminate\Foundation\Testing\WithoutMiddleware;
use Tests\TestCase;

class ExampleTest extends TestCase
{
use RefreshDatabase;

/**
* A basic functional test example.
*
* @return void
*/
public function test_basic_example()
{
$response = $this->get('/');

// ...
}
}

Illuminate\Foundation\Testing\RefreshDatabase 트레이트는 데이터베이스의 스키마가 최신 상태라면 마이그레이션을 다시 수행하지 않습니다. 대신 데이터베이스 트랜잭션 내에서 테스트를 실행합니다. 즉, 이 트레이트를 사용하지 않는 테스트 케이스에서 추가된 레코드는 데이터베이스에 남아있을 수 있습니다.

만약 데이터베이스를 완전히 초기 상태로 되돌리고 싶다면, Illuminate\Foundation\Testing\DatabaseMigrations 또는 Illuminate\Foundation\Testing\DatabaseTruncation 트레이트를 사용할 수 있습니다. 그러나 두 방법 모두 RefreshDatabase 트레이트보다는 처리 속도가 훨씬 느립니다.

모델 팩토리

테스트를 진행할 때 테스트 실행 전에 데이터베이스에 몇 개의 레코드를 삽입해야 할 때가 있습니다. 이때 일일이 각 컬럼의 값을 지정해서 테스트 데이터를 만들 필요 없이, 라라벨의 모델 팩토리를 이용하면 각 Eloquent 모델에 대해 기본 속성(attribute) 세트를 미리 정의하여 손쉽게 테스트 데이터를 생성할 수 있습니다.

모델 팩토리의 생성 및 활용 방법은 모델 팩토리 공식 문서를 참고해 주세요. 팩토리를 정의해두었다면, 테스트 내에서 다음과 같이 팩토리를 활용해 모델을 생성할 수 있습니다:

use App\Models\User;

public function test_models_can_be_instantiated()
{
$user = User::factory()->create();

// ...
}

시더 실행

기능 테스트(Feature Test) 도중에 데이터베이스 시더를 사용하여 데이터베이스를 채우고 싶다면, seed 메서드를 호출하면 됩니다. 기본적으로 seed 메서드는 DatabaseSeeder를 실행하며, 이를 통해 여러분이 정의한 모든 시더가 실행됩니다. 원한다면, 특정 시더 클래스명을 seed 메서드에 파라미터로 전달하여 일부 시더만 실행할 수도 있습니다:

<?php

namespace Tests\Feature;

use Database\Seeders\OrderStatusSeeder;
use Database\Seeders\TransactionStatusSeeder;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Illuminate\Foundation\Testing\WithoutMiddleware;
use Tests\TestCase;

class ExampleTest extends TestCase
{
use RefreshDatabase;

/**
* Test creating a new order.
*
* @return void
*/
public function test_orders_can_be_created()
{
// DatabaseSeeder 실행...
$this->seed();

// 특정 시더 실행...
$this->seed(OrderStatusSeeder::class);

// ...

// 여러 시더를 한 번에 실행...
$this->seed([
OrderStatusSeeder::class,
TransactionStatusSeeder::class,
// ...
]);
}
}

또한, RefreshDatabase 트레이트를 사용하는 각 테스트 시작 전마다 라라벨이 자동으로 시더를 실행하도록 할 수도 있습니다. 이 기능은 기본 테스트 클래스에 $seed 속성을 정의하여 활성화할 수 있습니다:

<?php

namespace Tests;

use Illuminate\Foundation\Testing\TestCase as BaseTestCase;

abstract class TestCase extends BaseTestCase
{
use CreatesApplication;

/**
* Indicates whether the default seeder should run before each test.
*
* @var bool
*/
protected $seed = true;
}

$seed 속성이 true일 경우, RefreshDatabase 트레이트를 사용하는 각 테스트 전에 Database\Seeders\DatabaseSeeder 클래스가 실행됩니다. 만약 매번 실행할 특정 시더를 지정하고 싶다면, 테스트 클래스에서 $seeder 속성을 정의하면 됩니다:

use Database\Seeders\OrderStatusSeeder;

/**
* Run a specific seeder before each test.
*
* @var string
*/
protected $seeder = OrderStatusSeeder::class;

사용 가능한 Assertion

라라벨은 PHPUnit 기능 테스트에서 사용할 수 있는 다양한 데이터베이스 assertion(검증 메서드)을 제공합니다. 아래에서 각각의 assertion에 대해 설명합니다.

assertDatabaseCount

데이터베이스의 특정 테이블에 지정한 개수만큼의 레코드가 존재하는지 검증합니다:

$this->assertDatabaseCount('users', 5);

assertDatabaseHas

데이터베이스의 특정 테이블에서 주어진 키/값 조건과 일치하는 레코드가 존재하는지 검증합니다:

$this->assertDatabaseHas('users', [
'email' => '[email protected]',
]);

assertDatabaseMissing

데이터베이스의 특정 테이블에 주어진 키/값 조건과 일치하는 레코드가 존재하지 않는지 검증합니다:

$this->assertDatabaseMissing('users', [
'email' => '[email protected]',
]);

assertSoftDeleted

assertSoftDeleted 메서드는 지정한 Eloquent 모델이 '소프트 삭제' 상태임을 확인할 때 사용합니다:

$this->assertSoftDeleted($user);

assertNotSoftDeleted

assertNotSoftDeleted 메서드는 지정한 Eloquent 모델이 '소프트 삭제' 상태가 아님을 확인할 때 사용합니다:

$this->assertNotSoftDeleted($user);

assertModelExists

지정한 모델 인스턴스가 데이터베이스에 존재하는지 검증합니다:

use App\Models\User;

$user = User::factory()->create();

$this->assertModelExists($user);

assertModelMissing

지정한 모델 인스턴스가 데이터베이스에 존재하지 않는지 검증합니다:

use App\Models\User;

$user = User::factory()->create();

$user->delete();

$this->assertModelMissing($user);

expectsDatabaseQueryCount

expectsDatabaseQueryCount 메서드는 테스트가 시작할 때, 해당 테스트 내에서 실행될 데이터베이스 쿼리의 총 개수를 미리 기대값으로 지정할 수 있습니다. 실제 쿼리 실행 횟수가 기대값과 정확히 일치하지 않으면 테스트가 실패합니다:

$this->expectsDatabaseQueryCount(5);

// Test...