【问题标题】:Mocking The PDO Object using PHPUnit使用 PHPUnit 模拟 PDO 对象
【发布时间】:2011-03-09 12:17:12
【问题描述】:

我在用 PHPUnit 模拟 PDO 对象时遇到了困难。

网络上似乎没有太多关于我的问题的信息,但我可以收集到的信息:

  1. PDO 具有“最终”__wakeup 和 __sleep 防止它被序列化的方法。
  2. PHPunit 的模拟对象实现会在某些时候序列化对象。
  3. 发生这种情况时,单元测试会因 PDO 生成的 PHP 错误而失败。

通过在单元测试中添加以下行来防止这种行为:

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 专业人士帮我一把?

谢谢,

【问题讨论】:

  • 有趣的是人们如何调试他们的代码。我更喜欢直接调试代码并将所有这些单元测试垃圾扔进垃圾箱。我把它留给需要自动化测试的大人物,因为他们的应用程序太大了,它可能会在无人火星任务中自行起飞。

标签: php unit-testing pdo phpunit


【解决方案1】:

我能想到的最好的方法是使用runkit 并使用 runkit_function_redefine 将两个最终方法重新定义为受保护的。

不要在 php.ini 中启用 runkit.internal_override 设置。

和以往一样,与 eval 一样,如果 runkit 似乎是答案,那么问题可能是错误的 :)

【讨论】:

  • 我不认为将runkiteval 用于测试目的有什么问题。
【解决方案2】:

$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']
          )
        );
    }

当您要求 getMock 不执行原始构造函数时,会使用这种带有反序列化的“技巧”,它会立即因 PDO 失败。

那么,如何解决呢?

一种选择是创建这样的测试助手

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

}

这里的目标是摆脱您不需要的原始 PDO 构造函数。然后,将您的测试代码更改为:

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

像这样创建模拟执行原始构造函数,但是由于有了 mockPDO 测试助手,它现在是无害的,您可以继续测试。

【讨论】:

  • 你是爸爸!非常感谢这工作正常。我已经放弃解决这个问题了!
  • 我遇到了与原始发帖人相同的问题,并使用了您的解决方案。但是,现在我的打字提示不再将其视为 PDO。 must be an instance of PDO, instance of Mock_PDOMock_96936f72 given
  • @nvanesch 它似乎也不算是instanceof \PDO。我最终做的只是直接实例化 mockPDO 类。
【解决方案3】:

你是在你的测试用例中实例化你的测试用例吗?

$classToTest = new MyTest($pdoMock);

现在,您实际上是在测试您的测试用例。它应该更像:

$classToTest = new My($pdoMock);

【讨论】:

  • 这绝对是原始问题中的一个错误。
猜你喜欢
  • 2015-11-01
  • 1970-01-01
  • 2014-08-23
  • 2021-10-31
  • 2012-09-26
  • 1970-01-01
  • 2011-07-19
  • 2012-05-05
  • 1970-01-01
相关资源
最近更新 更多