在Laravel中设置PHPUnit测试

时间:2015-11-17 10:03:11

标签: php laravel phpunit laravel-5.1

我对单元测试相当新,但我已经阅读了phpunit.de上的所有文档(截至第10章)。

它声明使用数据库进行测试可能会很慢,但如果设置正确,则可以与非数据库测试一样快。

因此,我想在Laravel中测试一个模型。我已经创建了一个模型工厂来将数据植入数据库。

我也创建了一个基本测试。

在PHPUnits文档中,它指出在每次测试之前,调用setUp()方法来设置测试。还有另一种静态方法setUpBeforeClass()

我想只为我的数据库表播种一次,并使用我测试中的记录。所以我使用Laravels factory()函数从setUpBeforeClass()方法中为数据库设定种子。

这是我的代码:

class CommentTest extends TestCase
{
    protected static $blog;
    protected static $comments;

    public static function setUpBeforeClass()
    {
        parent::setUpBeforeClass();

        self::$blog = factory(App\Models\Content\Blog::class)->create();
        self::$comments = factory(App\Models\Content\Comment::class, 6)->create();
    }

    public function testSomething()
    {
        $this->assertTrue(true);
    }
}

但是,当我运行phpunit时,我收到以下错误:

Fatal error: Call to a member function make() on a non-object in \vendor\laravel\framework\src\Illuminate\Foundation\helpers.php on line 54

Call Stack:
    0.0002     240752   1. {main}() \vendor\phpunit\phpunit\phpunit:0
    0.0173    1168632   2. PHPUnit_TextUI_Command::main() \vendor\phpunit\phpunit\phpunit:47
    0.0173    1175304   3. PHPUnit_TextUI_Command->run() \vendor\phpunit\phpunit\src\TextUI\Command.php:100
    2.9397    5869416   4. PHPUnit_TextUI_TestRunner->doRun() \vendor\phpunit\phpunit\src\TextUI\Command.php:149
    2.9447    6077272   5. PHPUnit_Framework_TestSuite->run() \vendor\phpunit\phpunit\src\TextUI\TestRunner.php:440
    2.9459    6092880   6. PHPUnit_Framework_TestSuite->run() \vendor\phpunit\phpunit\src\Framework\TestSuite.php:747
    2.9555    6096160   7. call_user_func:{\vendor\phpunit\phpunit\src\Framework\TestSuite.php:697}() \vendor\phpunit\phpunit\src\Framework\TestSuite.php:697
    2.9555    6096272   8. CommentTest::setUpBeforeClass() \vendor\phpunit\phpunit\src\Framework\TestSuite.php:697
    2.9555    6096480   9. factory() \tests\CommentTest.php:18
    2.9556    6096656  10. app() \vendor\laravel\framework\src\Illuminate\Foundation\helpers.php:350

如果我将代码从setUpBeforeClass()移动到setUp()并运行它,它会按预期工作,但这肯定会因为每次测试为数据库播种而效率低下吗?

我的问题:

  1. 是否正在以setUpBeforeClass()正确的方式播种数据库?
  2. 如果是(问题1),为什么我在运行phpunit时遇到致命错误,在调用factory()之前我还有什么要做的吗?
  3. 如果我必须将代码放在setUp()方法中,是否会出现性能问题?
  4. 我是否应该使用setUpBeforeClass()setUp()方法播种?在Laravels文档中,它显示了在测试中发生种子的示例,但如果我正在运行100次测试(例如),播种100次是个好主意吗?

1 个答案:

答案 0 :(得分:14)

好的,经过一些调查(类)后,我确定在调用静态setUpBeforeClass()方法时尚未创建Laravel应用程序。

Laravel容器是在setUp()中第一次调用\vendor\laravel\framework\src\illuminate\Foundation\Testing\TestCase.php时创建的。这就是我将代码移动到setUp()方法时工作正常的原因。

然后将容器存储在$app中存储的\vendor\laravel\framework\src\illuminate\Foundation\Testing\ApplicationTrait.php属性中。

我可以通过将此代码添加到setUpBeforeClass()方法来手动创建容器实例:

$app = require __DIR__.'/../bootstrap/app.php';
$app->make(Illuminate\Contracts\Console\Kernel::class)->bootstrap();

但是这种方法看起来很不好看,我不喜欢它。

相反,我将种子代码移动到setUp()方法,但只有在类属性为null时才播种数据库。因此,它只会在setUp()的第一次调用中播种。任何后续电话都不会播种:

class CommentTest extends TestCase
{
    use DatabaseMigrations;

    protected static $blog;
    protected static $comments;

    public function setUp()
    {
        parent::setUp();

        $this->runDatabaseMigrations();

        if (is_null(self::$blog)) {
            self::$blog = factory(App\Models\Content\Blog::class, 1)->create();
            self::$comments = factory(App\Models\Content\Comment::class, 6)->create();
        }
    }
}

结合Laravels DatabaseMigrations特征进行测试,现在是工作流程:

  1. 称为PHPUnit
  2. 调用Test类,其中包含DatabaseMigrations trait
  3. 迁移数据库(创建表)
  4. 第一次调用setUp()方法,该方法将相关表格与测试数据相结合
  5. 运行测试,并访问测试数据
  6. 没有调用tearDown()方法,而DatabaseMigrations特征只是重置数据库,因此我的测试不必担心清理测试数据。
  7. 修改

    此外,似乎(虽然我不是100%),如果您使用自定义setUp()方法,则需要从被覆盖的runDatabaseMigrations()方法中手动调用setUp()

    public function setUp()
    {
        parent::setUp();
        $this->runDatabaseMigrations();
    
        /** Rest of Setup **/
    }
    
    如果重载runDatabaseMigrations()方法,

    setUp()似乎无法自动调用。

    我希望这会有所帮助,但如果其他人有更好的解决方案,请随时告诉我:)