Издевательский объект PDO с использованием PHPUnit

У меня возникли сложности издеваться над объектом PDO с помощью PHPUnit.

В Интернете не так много информации о моей проблеме, но из того, что я могу собрать:

  • PDO имеет "окончательный" __wakeup и __sleep методы, которые не позволяют сериализовать.
  • PHPunit mock object реализует сериализацию объекта в какой-то момент.
  • Тестирование модулей затем завершится неудачей с ошибкой PHP, созданной PDO, когда это произойдет.

Существует функция, предназначенная для предотвращения этого поведения, добавив следующую строку к вашему unit test:

class MyTest extends PHPUnit_Framework_TestCase

{    
    protected $backupGlobals = FALSE;
     // ...

}

Источник: http://sebastian-bergmann.de/archives/797-Global-Variables-and-PHPUnit.html

Это не работает для меня, мой тест по-прежнему вызывает ошибку.

Полный тестовый код:

class MyTest extends PHPUnit_Framework_TestCase
{

    /**
     * @var MyTest
     */
    private $MyTestr;

    protected $backupGlobals = FALSE;

    /**
     * Prepares the environment before running a test.
     */
    protected function setUp()
    {
        parent::setUp();

    }

    /**
     * Cleans up the environment after running a test.
     */
    protected function tearDown()
    {

        parent::tearDown();
    }

    public function __construct()
    {

        $this->backupGlobals = false;
        parent::__construct();

    }


    /**
     * Tests MyTest->__construct()
     */
    public function test__construct()
    {

        $pdoMock = $this->getMock('PDO', array('prepare'), array(), '', false);

        $classToTest = new MyTest($pdoMock);

        // Assert stuff here!


    }

    // More test code.......

Любой PHPUnit pro дает мне руку?

Спасибо,

Бен

Ответ 1

$backupGlobals не помогает, потому что эта ошибка возникает из других источников. PHPUnit 3.5.2 (возможно, более ранние версии) имеет следующий код в PHPUnit/Framework/MockObject/Generator.php

    if ($callOriginalConstructor &&
        !interface_exists($originalClassName, $callAutoload)) {
        if (count($arguments) == 0) {
            $mockObject = new $mock['mockClassName'];
        } else {
            $mockClass  = new ReflectionClass($mock['mockClassName']);
            $mockObject = $mockClass->newInstanceArgs($arguments);
        }
    } else {
        // Use a trick to create a new object of a class
        // without invoking its constructor.
        $mockObject = unserialize(
          sprintf(
            'O:%d:"%s":0:{}',
            strlen($mock['mockClassName']), $mock['mockClassName']
          )
        );
    }

Этот "трюк" с unserialize используется, когда вы просите getMock не выполнять оригинальный конструктор, и он быстро завершит с PDO.

Итак, как это работает?

Один из вариантов заключается в создании тестового помощника, такого как

class mockPDO extends PDO
{
    public function __construct ()
    {}

}

Цель здесь - избавиться от исходного конструктора PDO, который вам не нужен. Затем измените свой тестовый код на это:

$pdoMock = $this->getMock('mockPDO', array('prepare'));

Создание макета, подобного этому, будет выполнять исходный конструктор, но поскольку теперь он безвреден благодаря помощнику проверки mockPDO, вы можете продолжить тестирование.

Ответ 2

Самое лучшее, о чем я могу думать, это использовать runkit и переопределить два последних метода, защищенных с помощью runkit_function_redefine.

Dont, чтобы включить параметр runkit.internal_override в php.ini.

И как всегда, как с eval, если runkit кажется ответом, вопрос, вероятно, неверен:)

Ответ 3

Вы создаете тестовый пример в тестовом случае?

$classToTest = new MyTest($pdoMock);

В настоящий момент вы, по сути, проверяете свой тестовый пример. Это должно быть нечто большее:

$classToTest = new My($pdoMock);