首页 文章

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

提问于
浏览
23

使用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是文件系统中第一个来自library / Cob的文件

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

5 回答

  • 0

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

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

    VS

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

    我发现当使用 incorrect @coversDefaultClass (同样可能适用于 @covers )时,我收到了此错误 . 在我的情况下,它是 \Path\To\\Class (双''),它正在搞乱要求调用 .

  • 0

    解决这个问题的一般答案:

    此错误消息通常指向自动加载和/或需要语句的问题 .

    在代码覆盖率报告生成期间,phpunit是 require ,每个 *.php 都在 <whitelist> 中 . 可以使用 addUncoveredFilesFromWhitelist=false 作为参数关闭此功能,但建议将其保持打开状态 .

    可能的原因

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

    其他原因可以

    • 类的重复定义(一个用于调试一个用于 生产环境 ,应该通过继承而不是通过加载正确的php文件来解决)

    • 不一致的大写导致PHP代码中的错误,如:

    if( !$classesLoaded['ThIsClass']) require_once(...); 在不同大写的多个地方 .

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

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

  • 0

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

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

相关问题