使用 @dataProvider 时出现“未找到类'Eloquent'”的PHPUnit问题。

7

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

PHP致命错误: 在路径为/path/to/project/app/models/ExampleClass.php的第7行找不到类'Eloquent'

看起来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;
}

示例测试类:

代码如下:

<?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

尝试使用 use \Eloquent;。一些 PHP 构建需要指向根命名空间。 - rdiz
@Dencker,不,那并没有帮助。编辑:删除use Eloquent;并使用class ExampleClass extends \Eloquent也没有帮助。 - JudRoman
5个回答

7
我正在遇到这个确切的问题。这与 Laravel 的别名(例如 Config::Log:: 等)在调用 @dataProvider 方法时尚未加载有关。这里有两种解决方案。

解决方案 1

修改你的 @dataProvider,使其不使用模型类。在我的情况下,我是在 @dataProvider 方法中创建模型对象,像这样:

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

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

    return [$person1, $person2];
}

由于@dataProvider方法引用了Person类,因此它将尝试加载该类。然后它将失败,因为Laravel尚未创建Eloquent类别名。

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

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

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

    // Assertions...
}

在您的情况下,这意味着返回 [['true']] 而不是 [[ExampleClass :: TRUE]]
解决方案2
我没有看到使用Eloquent类别名的强制性原因。事实上,我不知道它为什么存在(除了可能“看起来”更好?)。我在IRC频道中提出了这个问题,但没有得到回应...所以如果有理由在这里使用别名,那么我就不知道它是什么。
话虽如此,如果您的模型类扩展了基础的\Illuminate\Database\Eloquent\Model类,而不是Eloquent别名,则您的测试将按原样工作。
<?php
namespace Acme\Models;

use \Illuminate\Database\Eloquent\Model;

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

我基本上在发布问题之前就实现了你的“解决方案1”来解决它,但我更喜欢“解决方案2”。 - JudRoman
1
解决方案2更好!谢谢! - serghei

1
你需要在dataProvider中运行此代码。正如Thomas Kelley所说,这是因为它会在运行CreateApplication代码之前获取dataProvider数据,无论你是否在setUp方法中使用该trait或自己创建应用程序。这也可能会生成“x指定的数据提供程序无效”的错误消息。如果只运行单个测试,PHPStorm将仅显示“未执行任何测试!”。
public function fooDataProvider() {
    $this->refreshApplication(); 
    //Code using laravel
}

0
发现了一个对我有用的小技巧 - 你可以在数据提供者数组中返回一个闭包,在测试中执行它。
闭包中的代码将在测试方法中执行,而不是在数据提供者函数中执行。
    #[DataProvider('provideMyData')]
    public function testSomething(Closure $getUser)
    {
        $user = $getUser();
        
        self::assertInstanceOf(UserModel::class, $user);
    }

    public function provideMyData(): array
    {
        return [
            [
                fn() => UserModel::create(['attributes']), // Arrow style syntax
            ],
            [
                function () { // Full function style
                    $user = new UserModel();
                    $user->name = 'Full Name';
                    $user->save();

                    return $user;
                },
            ],

        ];
    }

-1

我也遇到了同样的错误,原因是在我的

protected function setUp() : void {
}

tests/TestCase.php 文件中

我没有调用 parent::setUp()

当我调用它时,错误消失了。


-3

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

类似这样:

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

PHPUnit不会为您加载自动加载程序。


1
bootstrap="bootstrap/autoload.php" 不就是在 phpunit.xml 中的作用吗?那个文件执行了 require __DIR__.'/../vendor/autoload.php'; - JudRoman

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接