Невозможно переопределить ошибку класса при генерации отчета о покрытии кода PHPUnit
Запуск проекта с Zend Framework 1.10 и Doctrine 2 (Beta1). Я использую пространства имен в своем собственном библиотечном коде.
При генерации отчетов о покрытии кода я получаю Fatal Error об обновлении класса. Чтобы предоставить дополнительную информацию, я прокомментировал вызов xdebug_disable() в моем исполняемом файле phpunit, чтобы вы могли увидеть трассировку функции (отключены локальные переменные, потому что было слишком много вывода).
Здесь мой вывод терминала:
$ 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).
Ответы
Ответ 1
Общий ответ для завершения этой проблемы:
Это сообщение об ошибке обычно указывает на проблему с автоматической загрузкой и/или запросами.
Во время формирования отчета о покрытии кода phpunit составляет require
каждый *.php
, который находится в <whitelist>
. Это можно отключить, используя addUncoveredFilesFromWhitelist=false
как
, но рекомендуется сохранить это.
Возможные причины
В этих случаях обычно бывает, что один из файлов имеет оператор require
, который требует уже загруженного класса (так как это не require_once).
Другие причины могут быть
- дублирующее определение классов (одно для отладки для производства, должно решаться путем наследования не путем загрузки правого файла php)
-
Несогласованная капитализация, приводящая к ошибкам в php-коде, например:
if( !$classesLoaded['ThIsClass']) require_once(...);
в нескольких местах с разными заглавными буквами.
- Класс, который не загружается вообще в тестах, но макет-объект был создан с именем загруженного тогда класса. Они могут столкнуться и привести к этой ошибке.
Ответ 2
Еще одна вещь, которую нужно проверить (особенно если вы работаете с виртуальной машиной) - это то, был ли файл переименован, но все еще существует в исходной форме. например. fooMapper.php переименован в FooMapper.php(например, через обновление SVN), ваша среда IDE обнаруживает новый файл и отправляет его на виртуальную машину. Теперь у вас есть обе версии на вашей виртуальной машине, даже если ваша локальная сеть локально только показывает один. Осторожно.
Ответ 3
Быстрое обходное решение может заключаться в том, чтобы добавить оператор if в начале декларации вашего класса, чтобы избежать повторной записи при запуске phpunit (если и только если это единственный класс, с которым вы столкнулись)
if (!class_exists("class_name")) {
// declare the class
class class_name
{
//...
}
}
Ответ 4
Это случилось со мной, потому что у меня была опечатка в объявлении @covers. Обратите внимание на дополнительную\в первой строке.
App\Controller\\Account\UsersController::forgotPassword()
VS
App\Controller\Account\UsersController::forgotPassword()
Ответ 5
Я обнаружил, что я получил эту ошибку, когда была использована неверная @coversDefaultClass
(что тоже относится к @covers
). В моем случае это было \Path\To\\Class
(с двойным "\" ), которое возилось с требуемыми вызовами.