我对单元测试相当新,但我已经阅读了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()
并运行它,它会按预期工作,但这肯定会因为每次测试为数据库播种而效率低下吗?
我的问题:
setUpBeforeClass()
正确的方式播种数据库?factory()
之前我还有什么要做的吗?setUp()
方法中,是否会出现性能问题?setUpBeforeClass()
或setUp()
方法播种?在Laravels文档中,它显示了在测试中发生种子的示例,但如果我正在运行100次测试(例如),播种100次是个好主意吗?答案 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
特征进行测试,现在是工作流程:
DatabaseMigrations
trait setUp()
方法,该方法将相关表格与测试数据相结合tearDown()
方法,而DatabaseMigrations
特征只是重置数据库,因此我的测试不必担心清理测试数据。修改强>
此外,似乎(虽然我不是100%),如果您使用自定义setUp()
方法,则需要从被覆盖的runDatabaseMigrations()
方法中手动调用setUp()
:
public function setUp()
{
parent::setUp();
$this->runDatabaseMigrations();
/** Rest of Setup **/
}
如果重载runDatabaseMigrations()
方法, setUp()
似乎无法自动调用。
我希望这会有所帮助,但如果其他人有更好的解决方案,请随时告诉我:)