生成PHPUnit代码覆盖率报告时无法重新声明类错误

时间:2010-05-12 05:01:25

标签: php zend-framework code-coverage phpunit

使用Zend Framework 1.10和Doctrine 2(Beta1)启动项目。我在自己的库代码中使用名称空间。

生成代码覆盖率报告时,我收到有关重新声明类的致命错误。为了提供更多信息,我在phpunit可执行文件中注释了xdebug_disable()调用,这样你就可以看到函数trace(因为输出太多而禁用了局部变量输出)。

这是我的终端输出:

$ phpunit
PHPUnit 3.4.12 by Sebastian Bergmann.

........

Time: 4 seconds, Memory: 16.50Mb

OK (8 tests, 14 assertions)

Generating code coverage report, this may take a moment.PHP Fatal error:  Cannot redeclare class Cob\Application\Resource\HelperBroker in /Users/Cobby/Sites/project/trunk/code/library/Cob/Application/Resource/HelperBroker.php on line 93
PHP Stack trace:
PHP   1. {main}() /usr/local/zend/bin/phpunit:0
PHP   2. PHPUnit_TextUI_Command::main() /usr/local/zend/bin/phpunit:54
PHP   3. PHPUnit_TextUI_Command->run() /usr/local/zend/share/pear/PHPUnit/TextUI/Command.php:146
PHP   4. PHPUnit_TextUI_TestRunner->doRun() /usr/local/zend/share/pear/PHPUnit/TextUI/Command.php:213
PHP   5. PHPUnit_Util_Report::render() /usr/local/zend/share/pear/PHPUnit/TextUI/TestRunner.php:478
PHP   6. PHPUnit_Framework_TestResult->getCodeCoverageInformation() /usr/local/zend/share/pear/PHPUnit/Util/Report.php:97
PHP   7. PHPUnit_Util_Filter::getFilteredCodeCoverage() /usr/local/zend/share/pear/PHPUnit/Framework/TestResult.php:623

Fatal error: Cannot redeclare class Cob\Application\Resource\HelperBroker in /Users/Cobby/Sites/project/trunk/code/library/Cob/Application/Resource/HelperBroker.php on line 93

Call Stack:
    0.0004     322888   1. {main}() /usr/local/zend/bin/phpunit:0
    0.0816    4114628   2. PHPUnit_TextUI_Command::main() /usr/local/zend/bin/phpunit:54
    0.0817    4114964   3. PHPUnit_TextUI_Command->run() /usr/local/zend/share/pear/PHPUnit/TextUI/Command.php:146
    0.1151    5435528   4. PHPUnit_TextUI_TestRunner->doRun() /usr/local/zend/share/pear/PHPUnit/TextUI/Command.php:213
    4.2931   16690760   5. PHPUnit_Util_Report::render() /usr/local/zend/share/pear/PHPUnit/TextUI/TestRunner.php:478
    4.2931   16691120   6. PHPUnit_Framework_TestResult->getCodeCoverageInformation() /usr/local/zend/share/pear/PHPUnit/Util/Report.php:97
    4.2931   16691148   7. PHPUnit_Util_Filter::getFilteredCodeCoverage() /usr/local/zend/share/pear/PHPUnit/Framework/TestResult.php:623

(我不知道为什么它会两次显示错误......?)

这是我的phpunit.xml:

<phpunit bootstrap="./code/tests/application/bootstrap.php" colors="true">
    <!-- bootstrap.php changes directory to trunk/code/tests,
    all paths below are relative to this directory. -->

    <testsuite name="My Promotions">
        <directory>./</directory>
    </testsuite>

    <filter>
        <whitelist>
            <directory suffix=".php">../application</directory>
            <directory suffix=".php">../library/Cob</directory>
            <exclude>
                    <!-- By adding the below line I can remove the error -->
                <file>../library/Cob/Application/Resource/HelperBroker.php</file>
                <directory suffix=".phtml">../application</directory>
                <directory suffix=".php">../application/doctrine</directory>
                <file>../application/Bootstrap.php</file>
                <directory suffix=".php">../library/Cob/Tools</directory>
            </exclude>
        </whitelist>
    </filter>

    <logging>
        <log type="junit" target="../../build/reports/tests/report.xml" />
        <log type="coverage-html" target="../../build/reports/coverage" charset="UTF-8"
            yui="true" highlight="true" lowUpperBound="50" highLowerBound="80" />
    </logging>
</phpunit>

我在哪个接缝中添加了一个标签来隐藏此问题。我确实有另一个应用程序资源,但它没有问题(另一个是Doctrine 2资源)。我不确定为什么它特定于这个类,我的整个库都是自动加载的,所以它们不是任何包含/需要调用的地方。我想应该注意的是HelperBroker是文件系统中第一个源自库/ Cob的文件

我在Snow Leopard上使用所有软件的最新/最新版本(Zend Server,Zend Framework,Doctrine 2 Beta1,Phing,PHPUnit,PEAR)。

5 个答案:

答案 0 :(得分:33)

解决此问题的一般答案:

此错误消息通常表示自动加载和/或需要声明时出现问题。

在代码覆盖率报告生成过程中,phpunit require *.php中的每个<whitelist>。可以使用addUncoveredFilesFromWhitelist=false作为a来关闭此功能 参数,但建议保持此状态。

可能的原因

在这些情况下往往会发生的情况是其中一个文件的require语句需要再次加载一个类(因为它不是require_once)。

其他原因可以是

  • 类的重复定义(一个用于调试一个用于生产,应该通过继承而不是通过加载正确的php文件来解决)
  • 不一致的大写导致PHP代码中的错误,如:

    在具有不同大小写的多个地方

    if( !$classesLoaded['ThIsClass']) require_once(...);

  • 在测试中根本没有加载类,但是使用当时加载的类的名称创建了一个模拟对象。那些可能会发生碰撞并导致错误

答案 1 :(得分:1)

一个快速的解决方法可能是在你的类声明的开头添加一个if语句,以避免在运行phpunit时重新声明(如果,并且只有当这是你遇到问题的唯一类时)

if (!class_exists("class_name")) {
    // declare the class
    class class_name
    {
        //...
    }
}

答案 2 :(得分:1)

要检查的另一件事(特别是如果您正在使用VM)是文件是否已重命名但仍以原始格式存在。例如。 fooMapper.php重命名为FooMapper.php(例如通过SVN更新),IDE会检测到新文件并将其发送到VM。您现在在VM上有两个版本,即使您的IDE本地只显示一个版本。当心。

答案 3 :(得分:0)

这件事发生在我身上,因为我的@covers宣言中有一个拼写错误。注意第一行中的额外\。

App\Controller\\Account\UsersController::forgotPassword()

VS

App\Controller\Account\UsersController::forgotPassword()

答案 4 :(得分:0)

我发现我在使用不正确的 @coversDefaultClass(同样可能适用于@covers)注释时收到此错误。在我的情况下,它是\Path\To\\Class(双重&#39; \&#39;),它正在搞乱要求电话。