Laravel Eloquent模型单元测试

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Laravel Eloquent模型单元测试相关的知识,希望对你有一定的参考价值。

我正在尝试编写一个测试用例,用于测试Laravel 4.2中两个Eloquent模型之间关系的关联和分离

这是我的测试用例:

class BookingStatusSchemaTest extends TestCase
{

  private $statusText = "Confirmed";
  private $bookingStub;
  private $statusStub;

  public function testMigrateService()
  {

    $this->createTestData();

    $booking = $this->bookingStub;
    $status = $this->statusStub;

    /**
     * Check that the booking has no status. OK
     */
    $this->assertNull($booking->status);

    /**
     * Check that status has no booking. OK
     */
    $this->assertEquals(count($status->bookings), 0);

    /**
     * Add a status to the booking. OK
     */
    $booking->status()->associate($this->statusStub);

    /**
     * Check that status has a booking. NOT OK - This gives error
     */
    $this->assertEquals(count($status->bookings), 1);

    /**
     * Check that the booking has a status. OK
     */
    $this->assertNotNull($booking->status);

    /**
     * Do NOT delete the status, just set the reference
     * to it to null.
     */
    $booking->status = null;

    /**
     * And check again. OK
     */
    $this->assertNull($booking->status);
  }

  private function createTestData()
  {

    $bookingStatus = BookingStatus::create([ 
        'status' => $this->statusText 
    ]);

    $booking = Booking::create([ ]);

    $this->bookingStub = $booking;
    $this->statusStub = $bookingStatus;

  }

}

当我执行它时,我得到:

There was 1 failure:

1) BookingStatusSchemaTest::testMigrateService
Failed asserting that 1 matches expected 0.

预订型号:

class Booking extends Eloquent {

  /**
  * A booking have a status
  */
  public function status()
  {
    return $this->belongsTo('BookingStatus');
  }

}

BookingStatus型号:

class BookingStatus extends Eloquent
{
  protected $table = 'booking_statuses';
  protected $guarded = [ 'id' ];
  protected $fillable = ['status'];

  /**
   * A booking status belongs to a booking
   */
  public function bookings()
  {
    return $this->hasMany('Booking');
  }

}

这是bookingstatus的迁移架构:

  Schema::create('booking_statuses', function(Blueprint $table)
  {
    $table->increments('id');
    $table->string('status');
    $table->timestamps();
  });

还有预订:

Schema::create('bookings', function(Blueprint $table)
{
  $table->increments('id');
  $table->unsignedInteger('booking_status_id')->nullable();
  $table->timestamps();
});

我需要添加/更改哪些能够在我的测试用例中验证关系?

答案

已经有一段时间了,我完全忘记了这个问题。由于OP仍然对它感兴趣,我将尝试以某种方式回答这个问题。

所以我假设实际的任务是:如何测试两个Eloquent模型之间的正确关系?

我认为是Adam Wathan首先建议放弃“单元测试”和“功能测试”以及“我不知道 - 这意味着什么测试”之类的术语,并将测试分成两个问题/概念:功能和单位,其中功能只是描述应用程序的功能,如“登录用户可以预订机票”,单位描述它的较低级别单位和它们公开的功能,如“预订有状态”。

我非常喜欢这种方法,考虑到这一点,我想重构你的测试:

class BookingStatusSchemaTest extends TestCase
{
    /** @test */
    public function a_booking_has_a_status()
    {
        // Create the world: there is a booking with an associated status
        $bookingStatus = BookingStatus::create(['status' => 'confirmed']);
        $booking = Booking::create(['booking_status_id' => $bookingStatus->id]);

        // Act: get the status of a booking
        $actualStatus = $booking->status;

        // Assert: Is the status I got the one I expected to get?
        $this->assertEquals($actualStatus->id, $bookingStatus->id);
    }


    /** @test */    
    public function the_status_of_a_booking_can_be_revoked()
    {
        // Create the world: there is a booking with an associated status
        $bookingStatus = BookingStatus::create(['status' => 'confirmed']);
        $booking = Booking::create(['booking_status_id' => $bookingStatus->id]);

        // Act: Revoke the status of a booking, e.g. set it to null
        $booking->revokeStatus();

        // Assert: The Status should be null now
        $this->assertNull($booking->status);
    }
}

此代码未经过测试!

请注意函数名称如何读取,如预订说明及其功能。您并不真正关心实施,您不必知道预订在何处或如何获得BookingStatus - 您只是想确保如果预订有BookingStatus,您可以获得BookingStatus。或者撤销它。或者也许改变它。或者做任何事情。您的测试显示了您希望如何与此单元进行交互。所以编写测试,然后尝试通过它。

你的测试中的主要缺陷可能是你有点“害怕”一些魔法发生。相反,将您的模型视为普通的旧php对象 - 因为它们就是这样!你不会在POPO上运行这样的测试:

/**
 * Do NOT delete the status, just set the reference
 * to it to null.
 */
$booking->status = null;

/**
 * And check again. OK
 */
$this->assertNull($booking->status);

这是一个非常广泛的主题,关于它的每一个声明都不可避免地被贬低。有一些指导方针可以帮助你相处,比如“只测试你自己的代码”,但很难把所有的和平放在一起。幸运的是,前面提到的Adam Wathan有一个非常优秀的视频课程,名为“Test Driven Laravel”,在那里他试驾了一个真实世界的Laravel应用程序。它可能有点贵,但它值得每一分钱,并帮助您了解测试方式比StackOverflow上的一些随机的家伙:)

另一答案

要测试您是否设置了正确的Eloquent关系,您必须针对关系类($model->relation())运行断言。你可以断言

  • 这是正确的关系类型断言$model->relation()HasManyBelongsToHasManyThrough等的实例
  • 它使用$model->relation()->getRelated()与正确的模型有关
  • 它使用$model->relation()->getForeignKey()使用正确的外键
  • 外键通过使用Schema::getColumListing($table)作为表中的列存在(这里,$table$model->relation()->getRelated()->getTable(),如果它是HasMany关系或$model->relation()->getParent()->getTable(),如果它是BelongsTo关系)

例如。假设你有一个Parent和一个Child模型,其中Parent通过使用Child作为外键的children()方法有许多parent_idParent映射parents表,Child映射children表。

$parent = new Parent;
# AppParent
$parent->children()
# IlluminateDatabaseEloquentRelationsHasMany
$parent->children()->getRelated()
# AppChild
$parent->children()->getForeignKey()
# 'parent_id'
$parent->children()->getRelated()->getTable()
# 'children'
Schema::getColumnListing($parent->children()->getRelated()->getTable())
# ['id', 'parent_id', 'col1', 'col2', ...]

编辑此外,这不会触及数据库,因为我们从不保存任何东西。但是,需要迁移数据库,否则模型不会与任何表关联。

以上是关于Laravel Eloquent模型单元测试的主要内容,如果未能解决你的问题,请参考以下文章

带有 Codeception 的 Laravel 4 模型单元测试 - 未找到“Eloquent”类

Laravel 5.1 Eloquent isFillable() 单元测试中的差异

如何模拟 Laravel Eloquent 模型的静态方法?

如何用 phpunit 模拟 Laravel Eloquent 模型?

在 Laravel(流明)上使用 Mockery 模拟 Eloquent 模型不起作用

Laravel:在 API 测试中使用 Eloquent 模型