PHPUnit MockObjects가 파라미터에 따라 다른 값을 반환하도록하려면 어떻게해야합니까?


141

'return value'인수에 상관없이 반환하는 PHPUnit 모의 객체가 있습니다 .

// From inside a test...
$mock = $this->getMock('myObject', 'methodToMock');
$mock->expects($this->any))
     ->method('methodToMock')
     ->will($this->returnValue('return value'));

내가 할 수있는 것은 mock 메소드에 전달 된 인수를 기반으로 다른 값을 반환하는 것입니다. 나는 다음과 같은 것을 시도했다.

$mock = $this->getMock('myObject', 'methodToMock');

// methodToMock('one')
$mock->expects($this->any))
     ->method('methodToMock')
     ->with($this->equalTo('one'))
     ->will($this->returnValue('method called with argument "one"'));

// methodToMock('two')
$mock->expects($this->any))
     ->method('methodToMock')
     ->with($this->equalTo('two'))
     ->will($this->returnValue('method called with argument "two"'));

그러나 이로 인해 모의 객체가 인수로 호출되지 않으면 PHPUnit이 불만을 제기 'two'하므로 methodToMock('two')정의가 첫 번째 정의를 덮어 쓴 것으로 가정합니다 .

그래서 내 질문은 : PHPUnit 모의 객체가 인수에 따라 다른 값을 반환하도록하는 방법이 있습니까? 그렇다면 어떻게?

답변:


125

콜백을 사용하십시오. 예 : (PHPUnit 문서와 직결) :

<?php
class StubTest extends PHPUnit_Framework_TestCase
{
    public function testReturnCallbackStub()
    {
        $stub = $this->getMock(
          'SomeClass', array('doSomething')
        );

        $stub->expects($this->any())
             ->method('doSomething')
             ->will($this->returnCallback('callback'));

        // $stub->doSomething() returns callback(...)
    }
}

function callback() {
    $args = func_get_args();
    // ...
}
?>

callback ()에서 원하는 처리를 수행하고 $ args에 따라 결과를 적절하게 반환하십시오.


2
설명서에 대한 링크를 제공 할 수 있습니까? "Google"
Kris Erickson

6
예를 들어 배열을 전달하여 메소드를 콜백으로 사용할 수 있습니다 $this->returnCallback(array('MyClassTest','myCallback')).
패트릭 피셔

1
클로저를 직접 전달하는 것도 가능해야합니다.
Ocramius

7
드문 경우에만 사용해야합니다. 콜백에 사용자 정의 논리를 작성할 필요가 없으므로 returnValueMap을 대신 사용하는 것이 좋습니다 .
Herman J. Radtke III

1
고마워요 또한 PHP 버전이 5.4보다 크면 익명 함수를 콜백으로 사용할 수 있습니다. $this->returnCallback(function() { // ... })
bmorenate

110

최신 phpUnit 문서에서 : "때때로 스텁 된 메소드는 사전 정의 된 인수 목록에 따라 다른 값을 리턴해야합니다. returnValueMap () 을 사용하여 인수를 해당 리턴 값과 연관시키는 맵을 작성할 수 있습니다 ."

$mock->expects($this->any())
    ->method('getConfigValue')
    ->will(
        $this->returnValueMap(
            array(
                array('firstparam', 'secondparam', 'retval'),
                array('modes', 'foo', array('Array', 'of', 'modes'))
            )
        )
    );

3
게시물의 링크가 오래되었습니다. 올바른 링크는 다음과 같습니다. returnValueMap ()
hejdav

49

비슷한 문제가있었습니다 (약간 다르지만 ... 인수에 따라 다른 반환 값이 필요하지 않지만 두 세트의 인수가 동일한 함수에 전달되는지 확인해야합니다). 나는 다음과 같은 것을 사용했다.

$mock = $this->getMock();
$mock->expects($this->at(0))
    ->method('foo')
    ->with(...)
    ->will($this->returnValue(...));

$mock->expects($this->at(1))
    ->method('foo')
    ->with(...)
    ->will($this->returnValue(...));

foo ()에 대한 두 번의 호출 순서를 알아야하기 때문에 완벽하지는 않지만 실제로는 그렇게 나쁘지 않습니다.


28

아마도 OOP 방식으로 콜백을 원할 것입니다.

<?php
class StubTest extends PHPUnit_Framework_TestCase
{
    public function testReturnAction()
    {
        $object = $this->getMock('class_name', array('method_to_mock'));
        $object->expects($this->any())
            ->method('method_to_mock')
            ->will($this->returnCallback(array($this, 'returnCallback'));

        $object->returnAction('param1');
        // assert what param1 should return here

        $object->returnAction('param2');
        // assert what param2 should return here
    }

    public function returnCallback()
    {
        $args = func_get_args();

        // process $args[0] here and return the data you want to mock
        return 'The parameter was ' . $args[0];
    }
}
?>

16

그것은 정확히 당신이 묻는 것이 아니지만 어떤 경우에는 도움이 될 수 있습니다.

$mock->expects( $this->any() ) )
 ->method( 'methodToMock' )
 ->will( $this->onConsecutiveCalls( 'one', 'two' ) );

onConsecutiveCalls- 지정된 순서로 값 목록을 리턴합니다.


4

각 요소가 다음의 배열 인 두 레벨 배열을 전달하십시오.

  • 첫 번째는 메소드 매개 변수이고 최소값은 리턴 값입니다.

예:

->willReturnMap([
    ['firstArg', 'secondArg', 'returnValue']
])

2

다음과 같이 인수를 반환 할 수도 있습니다.

$stub = $this->getMock(
  'SomeClass', array('doSomething')
);

$stub->expects($this->any())
     ->method('doSomething')
     ->will($this->returnArgument(0));

Mocking documentation 에서 볼 수 있듯이 메소드 returnValue($index)는 주어진 인수를 반환 할 수 있습니다.


0

이런 말이니?

public function TestSomeCondition($condition){
  $mockObj = $this->getMockObject();
  $mockObj->setReturnValue('yourMethod',$condition);
}

PHPUnit이 아닌 SimpleTest 코드라고 생각합니다. 그러나 아니오, 그것은 내가 달성하고자하는 것이 아닙니다. 주어진 숫자에 대한 단어를 반환하는 모의 객체가 있다고 가정 해보십시오. 내 mock 메소드는 1로 호출 될 때 "one"을 반환해야하고, 2 등으로 호출 할 때 "two"를 반환해야합니다. $
Ben Dowling

0

비슷한 문제가 발생하여 해결할 수 없었습니다 (PHPUnit에 대한 정보는 놀랍게도 거의 없습니다). 필자의 경우 각 테스트를 별도의 테스트 (알려진 입력 및 알려진 출력)로 만들었습니다. 나는 모든 거래 잭 모의 객체를 만들 필요가 없으며 특정 테스트를 위해 특정 객체 만 필요했기 때문에 테스트를 분리하고 코드의 개별 측면을 별도의 테스트 할 수 있습니다 단위. 이것이 귀하에게 해당되는지 여부는 확실하지 않지만 테스트해야 할 사항에 달려 있습니다.


불행히도 그것은 내 상황에서 작동하지 않을 것입니다. 모의는 테스트중인 메소드로 전달되고 테스트 메소드는 다른 인수로 모의 메소드를 호출합니다. 그래도 문제를 해결할 수 없다는 것이 흥미 롭습니다. 이것은 PHPUnit 제한 일 수 있습니다.
벤 Dowling

-1
$this->BusinessMock = $this->createMock('AppBundle\Entity\Business');

    public function testBusiness()
    {
        /*
            onConcecutiveCalls : Whether you want that the Stub returns differents values when it will be called .
        */
        $this->BusinessMock ->method('getEmployees')
                                ->will($this->onConsecutiveCalls(
                                            $this->returnArgument(0),
                                            $this->returnValue('employee')                                      
                                            )
                                      );
        // first call

        $this->assertInstanceOf( //$this->returnArgument(0),
                'argument',
                $this->BusinessMock->getEmployees()
                );
       // second call


        $this->assertEquals('employee',$this->BusinessMock->getEmployees()) 
      //$this->returnValue('employee'),


    }

-2

시도 :

->with($this->equalTo('one'),$this->equalTo('two))->will($this->returnValue('return value'));

이 답변은 원래 질문에는 적용되지 않지만 비슷한 문제를 자세히 설명합니다. 특정 매개 변수 집합 이 제공 되는지 확인하십시오 . PHPUnit의 with ()는 여러 개의 인수를 허용하며 각 매개 변수마다 하나씩 일치합니다.
TaZ
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.