在setupBeforeClass之后,有什么方法可以执行数据提供程序的功能?

时间:2018-08-30 10:28:18

标签: php phpunit phpunit-testing

我有一个单元测试类,在其中我想从另一个类实例化一个对象,以便使用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()

请帮助缓解此问题。

2 个答案:

答案 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会为您处理数据固定装置。