我有一个单元测试类,在其中我想从另一个类实例化一个对象,以便使用phpunit的setUpBeforeClass()固定装置。因此,如果我将直接在测试函数中使用最近实例化的对象,则它的工作正常。
如果我将这个对象用于为数据提供者创建的另一个函数中。因此,该对象设置为null会导致提供程序始终首先执行。
有没有一种方法可以在测试运行之前调用dataProvider?
require_once('Dashboard.php');
Class Someclass extends PHPUnit_Framework_TestCase {
protected static $_dashboard;
public static function setUpBeforeClass()
{
self::$_dashboard = new Dashboard();
self::$_dashboard->set_class_type('Member');
}
/**
* Test Org Thumb Image Existense
* param org profile image : array
* @dataProvider getOrgProfileImages
*/
public function testFieldValidation($a,$b){
//If I call that object function here it will give the result.
//$members = self::$_dashboard->get_members();
//var_dump($members); Printing result as expected
$this->assertTrue(true);
}
public function getOrgProfileImages() : array {
//var_dump(self::$_dashboard);
$members = self::$_dashboard->get_members();
$tmp_array = ['2','2'];
return $tmp_array;
}
public static function tearDownAfterClass()
{
self::$_dashboard = null;
}
}
错误: 为Someclass :: testFieldValidation指定的数据提供者无效。 在null上调用成员函数get_members()
请帮助缓解此问题。
答案 0 :(得分:0)
注意:由于我没有您的
Dashboard
类的来源,因此我在下面的示例中使用了随机数
在任何测试运行之前(以及在任何钩子,包括beforeClass
都有运行机会之前),会调用
提供程序。到目前为止,实现目标的最简单方法是在类加载时填充该静态属性:
use PHPUnit\Framework\TestCase;
/** @runTestsInSeparateProcesses enabled */
class SomeTest extends TestCase
{
public static $_rand = null;
public function provider()
{
$rand = self::$_rand;
var_dump(__METHOD__, getmypid(), 'provided rand', $rand);
return ['rand' => [$rand]];
}
/** @dataProvider provider */
public function testSomething($rand)
{
$this->expectNotToPerformAssertions();
var_dump(__METHOD__, getmypid(), 'tested with', $rand);
}
/** @dataProvider provider */
public function testSomethingElse($rand)
{
$this->expectNotToPerformAssertions();
var_dump(__METHOD__, getmypid(), 'tested with', $rand);
}
}
// this runs before anything happens to the test case class
// even before providers are invoked
SomeTest::$_rand = rand();
或者您可以在第一次调用时在提供程序中实例化仪表板:
public function provider()
{
// Instantiate once
if (null === self::$_rand) {
self::$_rand = rand();
}
$rand = self::$_rand;
var_dump(__METHOD__, getmypid(), 'provided rand', $rand);
return ['rand' => [$rand]];
}
答案 1 :(得分:0)
@ dirk-scholten是正确的。您应该为每个测试创建一个新对象。这是一个很好的测试实践。坦白地说,它看起来更像是您在测试数据而不是在测试代码,我想很好,这不是PHPUnit的典型用法。基于您要确保数据库中的每个用户都有缩略图的假设(只是猜测),我将采用以下方法:
<?php
class DashboardDataTest extends PHPUnit\Framework\TestCase {
private $dashboard;
public function setUp() {
$this->dashboard = new Dashboard();
}
/**
* Test Org Thumb Image Existence
* param org profile image : array
*
* @dataProvider getOrgProfileImages
*
* @param int $user_id
*/
public function testThumbnailImageExists(int $user_id){
$thumbnail = $this->dashboard->get_member_thumbnail($user_id);
$this->assertNotNull($thumbnail);
}
public function geOrgUserIDs() : array {
$dashboard = new Dashboard();
// Something that is slow
$user_ids = $dashboard->get_all_the_member_user_ids();
$data = [];
foreach($user_ids as $user_id){
$data[] = [$user_id];
}
return $data;
}
}
每个数据提供者将在测试之前被调用一次且仅被调用一次。您不需要类上的静态数据固定装置,因为在使用数据提供程序时phpunit会为您处理数据固定装置。