对Mock对象的PHPunit期望未得到满足

时间:2015-06-06 07:13:49

标签: php mocking phpunit expectations

这是我的班级:

public function __construct(Manager $moduleManager, Source\Yesno $yesNo)
{
    $this->moduleManager = $moduleManager;
    $this->yesNo = $yesNo;
}

public function my1()
{
    $this->moduleManager->isOutputEnabled('');
    $this->yesNo->toOptionArray();
}

public function my2()
{
    $this->moduleManager->isOutputEnabled('');
    $this->yesNo->toOptionArray();
}

这是我的测试:

...
    $this->observerMock = $this->getMock(
        'path\to\Observer',
        null,
        [$this->moduleManagerMock, $this->yesNoMock],
        '',
        true
    );
...
public function testMy1()
{
    $this->moduleManagerMock->expects($this->exactly(2))->method('isOutputEnabled');
    $this->yesNoMock->expects($this->exactly(2))->method('toOptionsArray');
    $this->observerMock->my1();
    $this->observerMock->my2();
}

测试返回:

  

方法名称的期望失败等于   当被调用2次时。预计方法被称为2次,   实际上叫了0次。

我的问题是:我几次面对这样的事情,但每次我都无法理解发生的事情。为什么第一个期望是正确的,第二个不合适?

Upd.1

我忘了说我几次遇到这样的情况。 这是我注意到的。使用xDebug我看到了内部测试

[this]
    [moduleManager] => ModuleManager_Mock_Name_<hash #1>
    [yesNo] => YesNo_Mock_Name_<hash #2>
    [observerObject] =>
        [moduleManager] => ModuleManager_Mock_Name_<hash #1>
        [yesNo] => YesNo_Mock_Name_<hash #3> <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<

moduleManager对象在unittest对象和内部观察者对象中具有相同的缓存。如果我在moduleMatcher上应用smth - 它会出现在两个地方

$ unittest-&gt; yes的哈希值与$ observerObject哈希值不同。如果我为某些方法设置匹配器 - 它仅适用于unittest类!

为什么会这样?如何防止创建不同的对象

Upd.2 FOUND!

通过对象管理器创建对象时

    $this->observerMock = $objectManager->getObject(
        'Observer',
        [
            'moduleManager' => $this->moduleManagerMock,
            'yesNo' => $this->yesNoMock,
        ]
    );

变量'moduleManager'和'yesNo'应该与构造函数中的变量相同:

public function __construct(Manager $moduleManager, Source\Yesno $yesNo)
{
    $this->moduleManager = $moduleManager;
    $this->yesNo = $yesNo;
}

以下是phpunit检查此代码的代码:

    foreach ($method->getParameters() as $parameter) {
        $parameterName = $parameter->getName();
        $argClassName = null;
        $defaultValue = null;

        if (array_key_exists($parameterName, $arguments)) {
            $constructArguments[$parameterName] = $arguments[$parameterName];
            continue;
        }

1 个答案:

答案 0 :(得分:1)

通常情况下,您不应该模拟/存储您正在测试的系统本身。因为在您的测试用例中,$this->observerMock对象本身就是一个存根对象(它模仿了另一个类的接口,但没有提供任何实现)。

这意味着方法m1m2也是模拟方法,它们在被调用时不会执行任何操作。随后,您的依赖项(moduleManagerMockyesNoMock)上的模拟方法将永远不会被称为,这就是您的期望失败的原因。

要正确测试您想要的行为,请直接使用Observer课程:

public function setUp() {
    $this->moduleManagerMock = $this->getMock(/*...*/);
    $this->yesNoMock = $this->getMock(/*...*/);

    // Do not generate a mock object of "Observer", but use the class 
    // under test itself!
    $this->observer = new Observer(
        $this->moduleManagerMock,
        $this->yesNoMock
    );
}

public function testM1() {
    $this->moduleManagerMock->expects($this->exactly(2))
                            ->method('isOutputEnabled');
    $this->yesNoMock->expects($this->exactly(2))
                    ->method('toOptionsArray');
    $this->observer->my1();
    $this->observer->my2();
}