数据库测试

未匹配的标注

数据库测试

介绍

Laravel 提供了各种有用的工具和断言,从而让测试数据库驱动变得更加容易。除此之外,Laravel 模型工厂和 Seeders 可以轻松地使用应用程序的 Eloquent 模型和关系来创建测试数据库记录。

每次测试后重置数据库

在进行测试之前,让我们讨论一下如何在每次测试后重置数据库,以便让先前测试的数据不会干扰后续测试。 Laravel 包含的 Illuminate\Foundation\Testing\RefreshDatabase trait 会为你解决这个问题。只需在您的测试类上使用该 Trait:

<?php

namespace Tests\Feature;

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

class ExampleTest extends TestCase
{
    use RefreshDatabase;

    /**
     * 一个基本的功能测试例子。
     */
    public function test_basic_example(): void
    {
        $response = $this->get('/');

        // ...
    }
}

如果你的数据库结构是最新的,那么这个 TraitIlluminate\Foundation\Testing\RefreshDatabase 并不会迁移数据库。相反,它只会在一个数据库事务中执行测试。因此,任何由测试用例添加到数据库的记录,如果不使用这个 Trait,可能仍然存在于数据库中。

如果你想使用迁移来完全重置数据库,可以使用这个 Trait Illuminate\Foundation\Testing\DatabaseMigrations 来替代。然而,DatabaseMigrations 这个 Trait 明显比 RefreshDatabase Trait 要慢。

模型工厂

当我们测试的时候,可能需要在执行测试之前向数据库插入一些数据。Laravel 允许你使用 模型工厂 为每个 Eloquent 模型 定义一组默认值,而不是在创建测试数据时手动指定每一列的值。

要学习有关创建和使用模型工厂来创建模型的更多信息,请参阅完整的 模型工厂文档。定义模型工厂后,你可以在测试中使用该工厂来创建模型:

use App\Models\User;

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

    // ...
}

运行 seeders

如果你在功能测试时希望使用 数据库 seeders 来填充你的数据库, 你可以调用 seed 方法。 默认情况下, seed 方法将会执行 DatabaseSeeder, 它将会执行你的所有其他 seeders。或者,你传递指定的 seeder 类名给 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;

    /**
     * 测试创建新订单。
     */
    public function test_orders_can_be_created(): void
    {
        // 运行 DatabaseSeeder...
        $this->seed();

        // 运行指定 seeder...
        $this->seed(OrderStatusSeeder::class);

        // ...

        // 运行指定的 seeders...
        $this->seed([
            OrderStatusSeeder::class,
            TransactionStatusSeeder::class,
            // ...
        ]);
    }
}

或者通过 RefreshDatabase trait 在每次测试之前自动为数据库填充数据。你也可以通过在测试类上定义 $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 trait 的测试之前运行 Database\Seeders\DatabaseSeeder 类。但是,你可以通过在测试类上定义 $seeder 属性来指定要执行的 seeder:

use Database\Seeders\OrderStatusSeeder;

/**
 * 在测试前指定要运行的 seeder
 *
 * @var string
 */
protected $seeder = OrderStatusSeeder::class;

可用的断言

Laravel 为你的 PHPUnit 功能测试提供了几个数据库断言。我们将在下面逐个讨论。

assertDatabaseCount

断言数据库中的表包含给定数量的记录:

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

assertDatabaseHas

断言数据库中的表包含给定键/值查询约束的记录:

$this->assertDatabaseHas('users', [
    'email' => 'sally@example.com',
]);

assertDatabaseMissing

断言数据库中的表不包含给定键/值查询约束的记录:

$this->assertDatabaseMissing('users', [
    'email' => 'sally@example.com',
]);

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...

本文章首发在 LearnKu.com 网站上。

本译文仅用于学习和交流目的,转载请务必注明文章译者、出处、和本文链接
我们的翻译工作遵照 CC 协议,如果我们的工作有侵犯到您的权益,请及时联系我们。

原文地址:https://learnku.com/docs/laravel/10.x/da...

译文地址:https://learnku.com/docs/laravel/10.x/da...

上一篇 下一篇
《L04 微信小程序从零到发布》
从小程序个人账户申请开始,带你一步步进行开发一个微信小程序,直到提交微信控制台上线发布。
《L03 构架 API 服务器》
你将学到如 RESTFul 设计风格、PostMan 的使用、OAuth 流程,JWT 概念及使用 和 API 开发相关的进阶知识。
贡献者:3
讨论数量: 0
发起讨论 只看当前版本


暂无话题~