我正在尝试使用PHPUnit为php应用程序实现类似Django的测试实用程序。就像Django一样,我的意思是在运行第一个测试之前从主数据库创建一个单独的测试数据库,并在运行最后一个测试后删除它。即使一次运行多个测试用例,也只需创建一次测试数据库。
为此,我采取了以下方法 -
我定义了一个自定义测试套件类,以便我可以在其设置和拆卸方法中编写用于创建和删除db的代码,然后使用此类运行测试,如下所示
$ phpunit MyTestSuite
MyTestSuite定义了一个名为suite
的静态方法,我只使用glob
并将测试添加到testsuite,如下所示
public static function suite() {
$suite = new MyTestSuite();
foreach (glob('./tests/*Test.php') as $tc) {
require_once $tc;
$suite->addTestSuite(basename($tc, '.php'));
}
return $suite;
}
所有测试用例类都从PHPUnit_Framework_TestCase
的子类扩展,此类的设置和拆除方法负责加载和清除json fixture文件中的初始数据。
现在不行。测试正在增加,我需要一次只运行一个选定的测试。但由于我已经使用测试套件加载测试,因此无法使用--filter选项。 这让我觉得这种方法可能不正确。
所以我的问题是,在运行第一个测试之前和运行最后一个测试之后做什么的正确方法是什么,而不管PHPUnit如何找到它们?
PS:我没有使用PHPUnit_Extensions_Database_TestCase,而是我自己创建,填充和删除数据库的实现。
答案 0 :(得分:32)
我最近遇到过需要解决同一问题的问题。我尝试使用自定义类的__destruct
方法Edorian's answer,但它似乎是在每次测试结束时运行,而不是在所有测试结束时运行。
我没有在我的bootstrap.php文件中使用特殊的类,而是在我的所有测试结束后使用PHP的register_shutdown_function
函数来处理数据库清理,它看起来效果很好。
这是我在bootstrap.php文件中的一个例子
register_shutdown_function(function(){
some_db_cleanup_methods();
});
答案 1 :(得分:17)
我的两个不使用"Test Suites"
的自发想法。一个就是在底部。
使用PHPUnits test listeners
你可以做一个
public function startTestSuite(PHPUnit_Framework_TestSuite $suite)
{
if($suite->getName() == "yourDBTests") { // set up db
}
public function endTestSuite(PHPUnit_Framework_TestSuite $suite)
{
if($suite->getName() == "yourDBTests") { // tear down db
}
您可以在xml配置文件中的测试套件中定义所有数据库测试,如in the docs
所示
<phpunit>
<testsuites>
<testsuite name="db">
<dir>/tests/db/</dir>
</testsuite>
<testsuite name="unit">
<dir>/tests/unit/</dir>
</testsuite>
</testsuites>
</phpunit>
使用phpunits引导程序文件,您可以创建一个创建数据库的类,并在进程结束时将其分解为自己的__destruct
方法。
将对象的引用放在某个全局范围内将确保对象仅在所有测试结束时被解构。 (正如@beanland所指出的那样:使用register_shutdown_function()会更有意义!)
http://www.phpunit.de/manual/3.2/en/organizing-test-suites.html显示:
<?php
class MySuite extends PHPUnit_Framework_TestSuite
{
public static function suite()
{
return new MySuite('MyTest');
}
protected function setUp()
{
print "\nMySuite::setUp()";
}
protected function tearDown()
{
print "\nMySuite::tearDown()";
}
}
class MyTest extends PHPUnit_Framework_TestCase
{
public function testWorks() {
$this->assertTrue(true);
}
}
这在PHPUnit 3.6中运行良好,可以在3.7中使用。它不在当前的文档中,因为“测试套件类”有些不赞成/不鼓励,但它们将会存在很长一段时间。
请注意,拆除和设置每个测试用例的整个数据库对于对抗测试间依赖性非常有用,但如果不在内存中运行测试(如sqlite内存),速度可能不值得它
答案 2 :(得分:1)
在2020年,@ edorian的做法已经deprecated:
/**
* @throws Exception
*
* @deprecated see https://github.com/sebastianbergmann/phpunit/issues/4039
*/
public function testSuiteLoaderClass(): string
{
///
}
现在,我们需要通过TestRunner使用extensions。将此代码添加到phpunit.xml
中:
<extensions>
<extension class="Tests\Extensions\Boot"/>
</extensions>
<testsuites>
...
</testsuites>
Tests/Extensions/Boot.php
:
<?php
namespace Tests\Extensions;
use PHPUnit\Runner\AfterLastTestHook;
use PHPUnit\Runner\BeforeFirstTestHook;
class Boot implements BeforeFirstTestHook, AfterLastTestHook
{
public function executeBeforeFirstTest(): void
{
// phpunit --testsuite Unit
echo sprintf("testsuite: %s\n", $this->getPhpUnitParam("testsuite"));
// phpunit --filter CreateCompanyTest
echo sprintf("filter: %s\n", $this->getPhpUnitParam("filter"));
echo "TODO: Implement executeBeforeFirstTest() method.";
}
public function executeAfterLastTest(): void
{
// TODO: Implement executeAfterLastTest() method.
}
/**
* @return string|null
*/
protected function getPhpUnitParam(string $paramName): ?string
{
global $argv;
$k = array_search("--$paramName", $argv);
if (!$k) return null;
return $argv[$k + 1];
}
}
纯PHP:
phpunit --testsuite Unit --filter CreateCompanyTest
Laravel:
php artisan test --testsuite Unit --filter CreateCompanyTest
输出:
PHPUnit 9.3.10 by Sebastian Bergmann and contributors.
testsuite: Unit
filter: CreateCompanyTest
TODO: Implement executeBeforeFirstTest() method.