Проверьте заголовки PHP с помощью PHPUnit

98

Я пытаюсь использовать PHPunit для тестирования класса, который выводит некоторые настраиваемые заголовки.

Проблема в том, что на моей машине это:

<?php

class HeadersTest extends PHPUnit_Framework_TestCase {

    public function testHeaders()
    {
        ob_start();

        header('Location: foo');
        $headers_list = headers_list();
        header_remove();

        ob_clean();

        $this->assertContains('Location: foo', $headers_list);
    }
}

или даже это:

<?php

class HeadersTest extends PHPUnit_Framework_TestCase {

    public function testHeaders()
    {
        ob_start();

        header('Location: foo');
        header_remove();

        ob_clean();
    }
}

вернуть эту ошибку:

name@host [~/test]# phpunit --verbose HeadersTest.php 
PHPUnit 3.6.10 by Sebastian Bergmann.

E

Time: 0 seconds, Memory: 2.25Mb

There was 1 error:

1) HeadersTest::testHeaders
Cannot modify header information - headers already sent by (output started at /usr/local/lib/php/PHPUnit/Util/Printer.php:173)

/test/HeadersTest.php:9

FAILURES!
Tests: 1, Assertions: 0, Errors: 1.

Это выглядит так, как будто перед запуском теста на терминал выводится что-то еще, даже если не включен другой файл и нет другого символа перед началом тега PHP. Может быть, что-то внутри PHPunit вызывает это?

В чем может быть проблема?

титул
источник
14
Просто хотел осветить это, если есть другие люди, заинтересованные в этом. headers_list () не работает при запуске PHPunit (который использует PHP CLI), но вместо этого работает xdebug_get_headers ().
название

Ответы:

124

Проблема в том, что PHPUnit будет печатать заголовок на экране, и в этот момент вы не можете добавить дополнительные заголовки.

Чтобы решить эту проблему, нужно запустить тест в изолированном процессе. Вот пример

<?php

class FooTest extends PHPUnit_Framework_TestCase
{
    /**
     * @runInSeparateProcess
     */
    public function testBar()
    {
        header('Location : http://foo.com');
    }
}

Это приведет к:

$ phpunit FooTest.php
PHPUnit 3.6.10 by Sebastian Bergmann.

.

Time: 1 second, Memory: 9.00Mb

OK (1 test, 0 assertions)

Ключ - аннотация @runInSeparateProcess.

Если вы используете PHPUnit ~ 4.1 или что-то в этом роде и получаете сообщение об ошибке:

PHP Fatal error:  Uncaught Error: Class 'PHPUnit_Util_Configuration' not found in -:378
Stack trace:
#0 {main}
  thrown in - on line 378

Fatal error: Uncaught Error: Class 'PHPUnit_Util_Configuration' not found in - on line 378

Error: Class 'PHPUnit_Util_Configuration' not found in - on line 378

Call Stack:
    0.0013     582512   1. {main}() -:0

Попробуйте добавить это в свой файл начальной загрузки, чтобы исправить это:

<?php
if (!defined('PHPUNIT_COMPOSER_INSTALL')) {
    define('PHPUNIT_COMPOSER_INSTALL', __DIR__ . '/path/to/composer/vendors/dir/autoload.php');
}
Сэм Хеннесси
источник
7
это вызывает ошибки в некоторых операторах define (). PHPUnit_Framework_Exception: Примечание: константа xyz уже определена
Минхаз 02
1
@mebjas Это звучит не связанно.
SamHennessy 02
4
Я получил это при применении:PHP Fatal error: Uncaught exception 'Exception' with message 'Serialization of 'SplFileInfo' is not allowed' in phar:///usr/local/bin/phpunit/phpunit/Util/GlobalState.php:211
t1gor 05
2
Это определенно лучший вариант решения проблемы. Оно работало завораживающе!
xarlymg89
2
Мне пришлось использовать xdebug_get_headers (), чтобы получить массив установленных заголовков. В моем случае глобальная функция headers_list () не работала.
Шалом Сэм
108

Хотя запуск теста в отдельном процессе действительно решает проблему, при запуске большого набора тестов возникают заметные накладные расходы.

Мое исправление заключалось в том, чтобы направить вывод phpunit на stderr, например:

phpunit --stderr <options>

Это должно решить проблему, а также означает, что вам не нужно создавать функцию-оболочку и заменять все вхождения в вашем коде.

Джон Кэрнс
источник
3
Гениально! Никаких изменений в моем коде, никаких отдельных процессов, и он работает.
Alexfernandez
но буду ли я видеть ошибки, которые я сделал, и вывод error_reporting (E_ALL) ??
spankmaster79 02
1
@ spankmaster79 да, просто перейдет к стандартной ошибке вашего терминала. По умолчанию большинство терминалов будут распечатывать стандартный вывод и стандартную ошибку вместе, но на самом деле это отдельные потоки.
Джон Кэрнс,
я становлюсь сделан !!! tnx для этого трюка, это имеет смысл, об ошибках следует сообщать в stderror !!! Tnx
th3n3rd
42
Вы можете добавить stderr="true"в свой phpunit.xml, чтобы сэкономить несколько нажатий клавиш.
tszming
10

В стороне: для меня headers_list()все время возвращалось 0 элементов. Я заметил комментарий @titel к вопросу и решил, что он заслуживает особого упоминания здесь:

Просто хотел осветить это, если есть другие люди, заинтересованные в этом. headers_list()не работает при запуске PHPunit (который использует PHP CLI), но xdebug_get_headers()работает вместо этого.

HTH

Мелле
источник
4

Как уже упоминалось в комментарии, я думаю, что это лучшее решение для определения processIsolation в файле конфигурации XML, например

     <?xml version="1.0" encoding="UTF-8"?>
     <phpunit
        processIsolation            = "true"
        // ... 
     >
     </phpunit>

Таким образом, вам не нужно передавать параметр --stderr, который может раздражать ваших коллег.

PepeNietnagel
источник
4
Вероятно, лучше определить его только для теста, который этого требует. Установка этого параметра для всех тестов просто замедлит выполнение тестов.
Ши
3

У меня было более радикальное решение, чтобы использовать его $_SESSIONвнутри моих протестированных / включенных файлов. Я отредактировал один из файлов PHPUnit в ../PHPUnit/Utils/Printer.php, чтобы "session_start();"перед командой «print $ buffer» был указан .

Это сработало для меня как шарм. Но я считаю, что пользовательское решение "joonty" на данный момент является лучшим из всех.

Серхио Абреу
источник
Вы отправили запрос на перенос в репозиторий? Думаю, это может быть обычная проблема.
t1gor 05
Спасибо за подсказку, я вызвал session_start () в моем phpunit bootstrap.php, и у меня он работает
bumperbox
0

Альтернативным решением для @runInSeparateProcess является указание опции --process -olated при запуске PHPUnit:

name@host [~/test]# phpunit --process-isolation HeadersTest.php

Это аналогично установке параметра processIsolation = "true" в phpunit.xml.

Это решение имеет те же преимущества / недостатки, что и указание опции --stderr, которая, однако, не сработала в моем случае. В принципе, никаких изменений кода не требуется, даже если производительность может снизиться из-за запуска каждого теста в отдельном процессе PHP.

AlexB
источник
Это может быть первым шагом в отслеживании причины и выполнении некоторых тестов, но для создания поддерживаемых тестов просто вставьте аннотацию непосредственно в тестовый файл. Чем меньше «специальных» параметров требуется из командной строки, тем проще будет поддерживать конфигурацию системы CI.
Ши
кто хоть раз понижал #fail. этот ответ верен как еще один вариант.
fb
0

Используйте параметр --stderr для получения заголовков из PHPUnit после ваших тестов.

phpunit --stderr
NSukonny
источник