У меня есть фиктивный объект PHPUnit, который возвращается 'return value'
независимо от аргументов:
// From inside a test...
$mock = $this->getMock('myObject', 'methodToMock');
$mock->expects($this->any))
->method('methodToMock')
->will($this->returnValue('return value'));
Я хочу иметь возможность возвращать другое значение на основе аргументов, переданных фиктивному методу. Я пробовал что-то вроде:
$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'
, поэтому я предполагаю, что определение of methodToMock('two')
заменяет определение первого.
Итак, мой вопрос: есть ли способ заставить фиктивный объект PHPUnit возвращать другое значение на основе его аргументов? И если да, то как?
php
unit-testing
mocking
phpunit
Бен Доулинг
источник
источник
$this->returnCallback(array('MyClassTest','myCallback'))
.$this->returnCallback(function() { // ... })
Из последних документов phpUnit: «Иногда метод с заглушкой должен возвращать разные значения в зависимости от заранее определенного списка аргументов. Вы можете использовать returnValueMap () для создания карты, которая связывает аргументы с соответствующими возвращаемыми значениями».
$mock->expects($this->any()) ->method('getConfigValue') ->will( $this->returnValueMap( array( array('firstparam', 'secondparam', 'retval'), array('modes', 'foo', array('Array', 'of', 'modes')) ) ) );
источник
У меня была аналогичная проблема (хотя и немного другая ... Мне не нужно было другое возвращаемое значение на основе аргументов, но мне пришлось протестировать, чтобы убедиться, что 2 набора аргументов передаются одной и той же функции). Я наткнулся на что-то вроде этого:
$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 (), но на практике это, вероятно, не так уж плохо.
источник
Вероятно, вы захотите выполнить обратный вызов в стиле ООП:
<?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]; } } ?>
источник
Это не совсем то, о чем вы спрашиваете, но в некоторых случаях это может помочь:
$mock->expects( $this->any() ) ) ->method( 'methodToMock' ) ->will( $this->onConsecutiveCalls( 'one', 'two' ) );
onConsecutiveCalls - возвращает список значений в указанном порядке
источник
Передайте двухуровневый массив, где каждый элемент является массивом:
пример:
->willReturnMap([ ['firstArg', 'secondArg', 'returnValue'] ])
источник
Вы также можете вернуть аргумент следующим образом:
$stub = $this->getMock( 'SomeClass', array('doSomething') ); $stub->expects($this->any()) ->method('doSomething') ->will($this->returnArgument(0));
Как видно из документации Mocking , метод
returnValue($index)
позволяет возвращать заданный аргумент.источник
Вы имеете в виду что-то подобное?
public function TestSomeCondition($condition){ $mockObj = $this->getMockObject(); $mockObj->setReturnValue('yourMethod',$condition); }
источник
У меня была аналогичная проблема, которую я тоже не мог решить (для PHPUnit информации на удивление мало). В моем случае я просто сделал каждый тест отдельным тестом - известный вход и известный выход. Я понял, что мне не нужно делать объект-имитатор на все руки, мне нужен только конкретный объект для конкретного теста, поэтому я разделил тесты и могу тестировать отдельные аспекты своего кода как отдельный Блок. Я не уверен, применимо ли это к вам или нет, но это зависит от того, что вам нужно проверить.
источник
$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'), }
источник
Пытаться :
->with($this->equalTo('one'),$this->equalTo('two))->will($this->returnValue('return value'));
источник