PHPUnit:"Class 'Eloquent' not found" 使用@dataProvider 时

PHPUnit: "Class 'Eloquent' not found" when using @dataProvider

我 运行 在 Laravel 应用程序中使用 @dataProvider 使用 PHPUnit 编写单元测试时遇到问题。我收到的错误是:

PHP Fatal error: Class 'Eloquent' not found in /path/to/project/app/models/ExampleClass.php on line 7

看起来 dataProvider 中使用的常量导致了致命错误。

composer.json:

"psr-4": {
    "Acme\Models\": "app/models"
}

phpunit.xml:

<?xml version="1.0" encoding="UTF-8"?>
<phpunit backupGlobals="false"
         backupStaticAttributes="false"
         bootstrap="bootstrap/autoload.php"
         colors="true"
         convertErrorsToExceptions="true"
         convertNoticesToExceptions="true"
         convertWarningsToExceptions="true"
         processIsolation="false"
         stopOnFailure="false"
         syntaxCheck="false"
>
    <testsuites>
        <testsuite name="Application Test Suite">
            <directory>./app/tests/</directory>
        </testsuite>
    </testsuites>
</phpunit>

示例模型:

<?php
namespace Acme\Models;

use Eloquent;

class ExampleClass extends Eloquent
{
    /**
     * @var bool
     */
    const TRUE = true;
}

实例测试class:

<?php

use Acme\Models\ExampleClass;

class ExampleClassTest extends TestCase
{
    /**
     * Example test.
     *
     * @param int $value
     * @return void
     * @dataProvider testExampleTestDataProvider
     */
    public function testExampleTest($value)
    {
        $this->assertTrue($value);
    }

    /**
     * Data provider for testExampleTest.
     *
     * @return array
     */
    public function testExampleTestDataProvider()
    {
        return array(
            array(ExampleClass::TRUE),
        );
    }
}

堆栈跟踪:

PHP Stack trace:
PHP   1. {main}() /usr/local/bin/phpunit:0
PHP   2. PHPUnit_TextUI_Command::main() /usr/local/bin/phpunit:612
PHP   3. PHPUnit_TextUI_Command->run() phar:///usr/local/bin/phpunit/phpunit/TextUI/Command.php:138
PHP   4. PHPUnit_TextUI_Command->handleArguments() phar:///usr/local/bin/phpunit/phpunit/TextUI/Command.php:148
PHP   5. PHPUnit_Util_Configuration->getTestSuiteConfiguration() phar:///usr/local/bin/phpunit/phpunit/TextUI/Command.php:696
PHP   6. PHPUnit_Util_Configuration->getTestSuite() phar:///usr/local/bin/phpunit/phpunit/Util/Configuration.php:837
PHP   7. PHPUnit_Framework_TestSuite->addTestFiles() phar:///usr/local/bin/phpunit/phpunit/Util/Configuration.php:924
PHP   8. PHPUnit_Framework_TestSuite->addTestFile() /path/to/project/vendor/phpunit/phpunit/src/Framework/TestSuite.php:400
PHP   9. PHPUnit_Framework_TestSuite->addTestSuite() /path/to/project/vendor/phpunit/phpunit/src/Framework/TestSuite.php:374
PHP  10. PHPUnit_Framework_TestSuite->__construct() /path/to/project/vendor/phpunit/phpunit/src/Framework/TestSuite.php:289
PHP  11. PHPUnit_Framework_TestSuite->addTestMethod() /path/to/project/vendor/phpunit/phpunit/src/Framework/TestSuite.php:188
PHP  12. PHPUnit_Framework_TestSuite::createTest() /path/to/project/vendor/phpunit/phpunit/src/Framework/TestSuite.php:842
PHP  13. PHPUnit_Util_Test::getProvidedData() /path/to/project/vendor/phpunit/phpunit/src/Framework/TestSuite.php:465
PHP  14. ReflectionMethod->invoke() /path/to/project/vendor/phpunit/phpunit/src/Util/Test.php:392
PHP  15. ExampleClassTest->testExampleTestDataProvider() /path/to/project/vendor/phpunit/phpunit/src/Util/Test.php:392
PHP  16. spl_autoload_call() /path/to/project/vendor/phpunit/phpunit/src/Util/Test.php:27
PHP  17. Composer\Autoload\ClassLoader->loadClass() /path/to/project/vendor/phpunit/phpunit/src/Util/Test.php:0
PHP  18. Composer\Autoload\includeFile() /path/to/project/vendor/composer/ClassLoader.php:278
PHP  19. include() /path/to/project/vendor/composer/ClassLoader.php:386

我遇到了这个问题。这与调用 @dataProvider 方法时尚未加载 Laravel 别名(例如 Config::Log::...)有关。这里我能想到的有两种解决方法。

解决方案 1

修改您的 @dataProvider,使其不使用模型 class。就我而言,我在@dataProvider 方法中创建模型对象,如下所示:

public function people() {
    $person1 = new Person();
    $person1->name = "me";

    $person2 = new Person();
    $person2->name = "you";

    return [$person1, $person2];
}

由于 Person class 在 @dataProvider 方法中被引用,它将尝试加载 class。然后它将失败,因为 Eloquent class 别名尚未由 Laravel 创建。

为了解决这个问题,我可以 return 数据,并在测试本身中创建实际的模型对象:

public function people() {
    return ["me", "you"];
}

public function testPerson($name) {
    $person = new Person();
    $person->name = $name;

    // Assertions...
}

在您的情况下,这意味着 returning [['true']],而不是 [[ExampleClass::TRUE]]

解决方案 2

我认为没有令人信服的理由在这里使用 Eloquent class 别名。事实上,我根本不知道它为什么存在(也许除了 "looks" 更好?)。我在 IRC 频道中提出了这个问题,但没有得到回应...所以如果有理由在这里使用别名,那我就不知道了。

也就是说,如果您的模型 class 扩展了基础 \Illuminate\Database\Eloquent\Model class 而不是 Eloquent 别名,那么您的测试将按原样开始工作。

<?php
namespace Acme\Models;

use \Illuminate\Database\Eloquent\Model;

class ExampleClass extends Model
{
    /**
     * @var bool
     */
    const TRUE = true;
}

您需要先包含自动加载器。

类似于:

require __DIR__.'/../vendor/autoload.php'; 

PHPUnit 不会为您加载自动加载器。

您必须 运行 在您的 dataProvider 中这样做。正如 Thomas Kelley 所说的那样,因为如果您使用该特征或在 setUp 方法中自行创建应用程序,它会在获得 运行 CreateApplication 代码以太之前获取 dataProvider 数据。这也可以生成 The data provider specified for x is invalid. 如果单个测试是 运行,PHPStorm 只会说 No tests executed!

public function fooDataProvider() {
    $this->refreshApplication(); 
    //Code using laravel
}

我遇到了同样的错误,这是因为在我的

protected function setUp() : void {
}

tests/TestCase.php

我没有打电话parent::setUp()

当我这样做时,错误消失了。