21

这试图为类的方法编写一个测试,该方法调用带有闭包的模拟方法。你将如何验证被调用的闭包?

我知道您可以断言该参数是Closure. 但是您将如何检查有关关闭的任何信息?

例如,您将如何验证传递的函数:

 class SUT {
     public function foo($bar) {
         $someFunction = function() { echo "I am an anonymous function"; };
         $bar->baz($someFunction);
     }
 }

 class SUTTest extends PHPUnit_Framework_TestCase {
     public function testFoo() {
         $mockBar = $this->getMockBuilder('Bar')
              ->setMethods(array('baz'))
              ->getMock();
         $mockBar->expects($this->once())
              ->method('baz')
              ->with( /** WHAT WOULD I ASSERT HERE? **/);

         $sut = new SUT();

         $sut->foo($mockBar);
     }
 }

您无法比较 PHP 中的两个闭包。PHPUnit中有没有办法执行传入的参数或以某种方式验证它?

4

3 回答 3

18

如果你想模拟一个匿名函数(回调),你可以用__invoke方法模拟一个类。例如:

$shouldBeCalled = $this->getMock(\stdClass::class, ['__invoke']);
$shouldBeCalled->expects($this->once())
    ->method('__invoke');

$someServiceYouAreTesting->testedMethod($shouldBeCalled);

如果您使用的是最新的 PHPUnit,则必须使用 mock builder 来解决问题:

$shouldBeCalled = $this->getMockBuilder(\stdClass::class)
    ->setMethods(['__invoke'])
    ->getMock();

$shouldBeCalled->expects($this->once())
    ->method('__invoke');

$someServiceYouAreTesting->testedMethod($shouldBeCalled);

您还可以为方法参数设置期望值或设置返回值,就像对任何其他方法一样:

$shouldBeCalled->expects($this->once())
    ->method('__invoke')
    ->with($this->equalTo(5))
    ->willReturn(15);
于 2015-11-20T14:17:59.650 回答
9

您的问题是您没有注入依赖项(闭包),这总是使单元测试变得更加困难,并且可能使隔离变得不可能。

将闭包注入SUT::foo()而不是在其中创建它,您会发现测试要容易得多。

这是我设计该方法的方式(请记住,我对您的真实代码一无所知,因此这对您来说可能实用,也可能不实用):

class SUT 
{
    public function foo($bar, $someFunction) 
    {
        $bar->baz($someFunction);
    }
}

class SUTTest extends PHPUnit_Framework_TestCase 
{
    public function testFoo() 
    {
        $someFunction = function() {};

        $mockBar = $this->getMockBuilder('Bar')
             ->setMethods(array('baz'))
             ->getMock();
        $mockBar->expects($this->once())
             ->method('baz')
             ->with($someFunction);

        $sut = new SUT();

        $sut->foo($mockBar, $someFunction);
    }
}
于 2013-10-08T18:39:10.247 回答
9

如果闭包内部有一些副作用SUT,可以在模拟调用后通过测试验证,请使用returnCallback提供另一个闭包以使用传递的参数调用并将其返回值返回给SUT. 这将允许您调用SUT's 闭包来引起副作用。

 class SUT {
     public function foo($bar) {
         $someFunction = function() { return 5 * 3; };
         return $bar->baz($someFunction);
     }
 }

 class SUTTest extends PHPUnit_Framework_TestCase {
     public function testFoo() {
         $mockBar = $this->getMockBuilder('Bar')
              ->setMethods(array('baz'))
              ->getMock();
         $mockBar->expects($this->once())
              ->method('baz')
              ->will($this->returnCallback(function ($someFunction) {
                  return $someFunction();
              }));

         $sut = new SUT();

         self::assertEquals(15, $sut->foo($mockBar));
     }
 }
于 2013-10-09T17:10:19.223 回答