Skip to content

Method Stubbing

The Phake::when() method is used to stub methods in Phake. As discussed in the introduction, stubbing allows an object method to be forced to return a particular value given a set of parameters. Similarly to Phake::verify(), Phake::when() accepts a mock object generated from Phake::mock() as its first parameter.

Imagine I was in the process of building the next great online shopping cart. The first thing any good shopping cart allows is to be able to add items. The most important thing I want to know from the shopping cart is how much money in merchandise is in there. So, I need to make myself a ShoppingCart class. I also am going to need some class to define my items. I am more worried about the money right now and because of that I am keenly aware that any item in a shopping cart is going to have a price. So I will just create an interface to represent those items called Item. Now take a minute to marvel at the creativity of those names. Great, now check out the initial definitions for my objects.

/**
 * An item that is going to make me rich.
 */
interface Item
{
    public function getPrice()
}

/**
 * A customer's cart that will contain items that are going to make me rich.
 */
class ShoppingCart
{
    private array $items = [];

    /**
     * Adds an item to the customer's order
     */
    public function addItem(Item $item): void
    {
        $this->items[] = $item;
    }

    /**
     * Returns the current sub total of the customer's order
     */
    public function getSubTotal()
    {
    }
}

So, I am furiously coding away at this fantastic new ShoppingCart class when I realize, I am doing it wrong! You see, a few years ago I went to this conference with a bunch of other geeky people to talk about how to make quality software. I am supposed to be writing unit tests. Here I am, a solid thirteen lines (not counting comments) of code into my awe inspiring new software and I haven't written a single test. I tell myself, "There's no better time to change than right now!" So I decide to start testing. After looking at the options I decide PHPUnit with this sweet new mock library called Phake is the way to go.

My first test is going to be for the currently unimplemented ShoppingCart::getSubTotal() method. I already have a pretty good idea of what this function is going to need to do. It will need to look at all of the items in the cart, retrieve their price, add it all together and return the result. So, in my test I know I am going to need a fixture that sets up a shopping cart with a few items added. Then I am going to need a test that calls ShoppingCart::getSubTotal() and asserts that it returns a value equal to the price of the items I added to the cart. One catch though, I don't have any concrete instances of an Item. I wasn't even planning on doing any of that until tomorrow. I really want to just focus on the ShoppingCart class. Never fear, this is why I decided to use Phake. I remember reading about how it will allow me to quickly create instance of my classes and interfaces that I can set up stubs for so that method calls return predictable values. This project is all coming together and I am really excited.

class ShoppingCartTest extends PHPUnit\Framework\TestCase
{
    public function testGetSub()
    {
        $item1 = Phake::mock(Item::class);
        $item2 = Phake::mock(Item::class);
        $item3 = Phake::mock(Item::class);

        Phake::when($item1)->getPrice()->thenReturn(100);
        Phake::when($item2)->getPrice()->thenReturn(200);
        Phake::when($item3)->getPrice()->thenReturn(300);

        $shoppingCart = new ShoppingCart();
        $shoppingCart->addItem($item1);
        $shoppingCart->addItem($item2);
        $shoppingCart->addItem($item3);

        $this->assertEquals(600, $shoppingCart->getSubTotal());
    }
}

My test here shows a very basic use of Phake for creating method stubs. I am creating three different mock implementations of the Item class. Then for each of those item classes, I am creating a stub using Phake::when() that will return 100, 200, and 300 respectively. I know my method that I am getting ready to implement will need to call those methods in order to calculate the total cost of the order.

My test is written so now it is time to see how it fails. I run it with phpunit and see the output below

$ phpunit ExampleTests/ShoppingCartTest.php
PHPUnit 9.5.4 by Sebastian Bergmann and contributors.

F                                                                   1 / 1 (100%)

Time: 00:00.012, Memory: 4.00 MB

There was 1 failure:

1) ShoppingCartTest::testGetSub
Failed asserting that null matches expected 600.

/home/mikel/Documents/Projects/Phake/tests/ShoppingCartTest.php:69

FAILURES!
Tests: 1, Assertions: 1, Failures: 1.

Now that I have a working (and I by working I mean breaking!) test it is time to look at the code necessary to make the test pass.

class ShoppingCart
{
    // I am cutting out the already seen code.
    // If you want to see it again look at the previous examples!

    /**
     * Returns the current sub total of the customer's order
     */
    public function getSubTotal()
    {
        $total = 0;

        foreach ($this->items as $item)
        {
            $total += $item->getPrice();
        }

        return $total;
    }
}

The code here is pretty simple. I am just iterating over the ShoppingCart::$item property, calling the Item::getPrice() method, and adding them all together. Now when I run phpunit, the tests were successful and I am getting off to a great start with my shopping cart.

$ phpunit ExampleTests/ShoppingCartTest.php
PHPUnit 9.5.4 by Sebastian Bergmann and contributors.

.                                                                   1 / 1 (100%)

Time: 00:00.011, Memory: 4.00 MB

OK (1 test, 1 assertion)

So, what is Phake doing here? Phake is providing us a predictable implementation of the Item::getPrice() method that we can use in our test. It helps me to ensure the when my test breaks I know exactly where it is breaking. I will not have to be worried that a bad implementation of Item::getPrice() is breaking my tests.

How Phake::when() Works

Internally Phake is doing quite a bit when this test runs. The three calls to Phake::mock() are creating three new classes that in this case each implement the Item interface. These new classes each define implementations of any method defined in the Item interface. If Item extended another interface, implementations of all of that parent's defined methods would be created as well. Each method being implemented in these new classes does a few different things. The first thing that it does is record the fact that the method was called and stores the parameters that were used to call it. The next significant thing it does is looks at the stub map for that mock object. The stub map is a map that associates answers to method matchers. An answer is what a mocked object will return when it is called. By default, a call to a mock object returns a static answer of NULL. We will discuss answers more in Answers. A method matcher has two parts. The first is the method name. The second is an array of arguments. The array of arguments will then contain various constraints that are applied to each argument to see if a given argument will match. The most common constraint is an equality constraint that will match loosely along the same lines as the double equals sign in PHP. We will talk about matchers more in Method Parameter Matchers.

When each mock object is initially created, its stub map will be empty. This means that any call to a method on a mock object is going to return a default answer of NULL. If you want your mock object's methods to return something else you must add answers to the stub map. The Phake::when() method allows you to map an answer to a method matcher for a given mock object. The mock object you want to add the mapping to is passed as the first parameter to Phake::when(). The Phake::when() method will then return a proxy that can be used add answers to your mock object's stub map. The answers are added by making method calls on the proxy just as you would on the mock object you are proxying. In the first example above you saw a call to Phake::when($this->item1)->getPrice(). The getPrice() call here was telling Phake that I am about to define a new answer that will be returned any time $this->item->getPrice() is called in my code. The call to $this->item->getPrice() returns another object that you can set the answer on using Phake's fluent api. In the example I called Phake::when($this->item1)->getPrice()->thenReturn(100). The thenReturn() method will bind a static answer to a matcher for getPrice() in the stub map for $this->item1.

Overwriting Existing Stubs

My shopping cart application is coming right along. I can add items and the total price seems to be accurate. However, while I was playing around with my new cart I noticed a very strange problem. I was playing around with the idea of allowing discounts to be applied to a cart as just additional items that would have a negative price. So while I am playing around with this idea I notice that the math isn't always adding up. If I start with an item that is $100 and then add a discount that is $81.40 I see that the total price isn't adding up to $18.60. This is definitely problematic After doing some further research, I realize I made a silly mistake. I am just using simple floats to calculate the costs. Floats are by nature inaccurate. Once you start using them in mathematical operations they start to show their inadequacy for precision. In keeping with the test driven method of creating code I need to create a unit test this flaw.

class ShoppingCartTest extends PHPUnit\Framework\TestCase
{
      public function testGetSub()
      {
          $item1 = Phake::mock(Item::class);
          $item2 = Phake::mock(Item::class);
          $item3 = Phake::mock(Item::class);

          Phake::when($item1)->getPrice()->thenReturn(100);
          Phake::when($item2)->getPrice()->thenReturn(200);
          Phake::when($item3)->getPrice()->thenReturn(300);

          $shoppingCart = new ShoppingCart();
          $shoppingCart->addItem($item1);
          $shoppingCart->addItem($item2);
          $shoppingCart->addItem($item3);

          $this->assertEquals(600, $shoppingCart->getSubTotal());
      }

      public function testGetSubTotalWithPrecision()
      {
          $item1 = Phake::mock(Item::class);
          $item2 = Phake::mock(Item::class);
          $item3 = Phake::mock(Item::class);

          Phake::when($item1)->getPrice()->thenReturn(100);
          Phake::when($item2)->getPrice()->thenReturn(-81.4);
          Phake::when($item3)->getPrice()->thenReturn(20);

          $shoppingCart = new ShoppingCart();
          $shoppingCart->addItem($item1);
          $shoppingCart->addItem($item2);
          $shoppingCart->addItem($item3);

          $this->assertEquals(38.6, $shoppingCart->getSubTotal());
      }
}

You can see that I added another test method that uses actual floats for some of the prices as opposed to round numbers. Now when I run my test suite I can see the result.

$ phpunit ExampleTests/ShoppingCartTest.php
PHPUnit 9.5.4 by Sebastian Bergmann and contributors.

..                                                                  2 / 2 (100%)

Time: 00:00.012, Memory: 4.00 MB

OK (2 tests, 2 assertions)

Once you confirmed that your implementation works, I want to discuss streamlining test cases with you. You will notice that the code in ShoppingCartTest::testGetSubTotalWithPrecision() contains almost all duplicate code when compared to ShoppingCartTest::testGetSub(). If I were to continue following this pattern of doing things I would eventually have tests that are difficult to maintain. Phake allows you to very easily override stubs. This is very important in helping you to reduce duplication in your tests and leads to tests that will be easier to maintain. To overwrite a previous stub you simply have to redefine it. I am going to change ShoppingCartTest::testGetSubTotalWithPrecision() to instead just redefine the getPrice() stubs.

class ShoppingCartTest extends PHPUnit\Framework\TestCase
{
    private ShoppingCart $shoppingCart;
    private Item $item1;
    private Item $item2;
    private Item $item3;

    public function setUp(): void
    {
        $this->item1 = Phake::mock(Item::class);
        $this->item2 = Phake::mock(Item::class);
        $this->item3 = Phake::mock(Item::class);

        Phake::when($this->item1)->getPrice()->thenReturn(100);
        Phake::when($this->item2)->getPrice()->thenReturn(200);
        Phake::when($this->item3)->getPrice()->thenReturn(300);

        $this->shoppingCart = new ShoppingCart();
        $this->shoppingCart->addItem($this->item1);
        $this->shoppingCart->addItem($this->item2);
        $this->shoppingCart->addItem($this->item3);
    }

    public function testGetSub()
    {
        $this->assertEquals(600, $this->shoppingCart->getSubTotal());
    }

    public function testGetSubTotalWithPrecision()
    {
        Phake::when($this->item1)->getPrice()->thenReturn(100);
        Phake::when($this->item2)->getPrice()->thenReturn(-81.4);
        Phake::when($this->item3)->getPrice()->thenReturn(20);

        $this->assertEquals(38.6, $this->shoppingCart->getSubTotal());
    }
}

If you rerun this test you will get the same results shown in before. The test itself is much simpler though there is much less unnecessary duplication. The reason this works is because the stub map I was referring to in How Phake::when() works isn't really a map at all. It is more of a stack in reality. When a new matcher and answer pair is added to a mock object, it is added to the top of the stack. Then whenever a stub method is called, the stack is checked from the top down to find the first matcher that matches the method that was called. So, when I created the additional stubs for the various Item::getPrice() calls, I was just adding additional matchers to the top of the stack that would always get matched first by virtue of the parameters all being the same.

Resetting A Mock's Stubs

If overriding a stub does not work for your particular case and you would rather start over with all default stubs then you can use Phake::reset() and Phake::resetStatic(). These will remove all stubs from a mock and also empty out all recorded calls against a mock. Phake::reset() will do this for instance methods on the mock and Phake::resetStatic() will do this for all static methods on the mock.

public function testResettingStubMapper()
{
    $mock = Phake::mock(PhakeTest_MockedClass::class);
    Phake::when($mock)->foo()->thenReturn(42);

    $this->assertEquals(42, $mock->foo());

    Phake::reset($mock);
    //$mock->foo() now returns the default stub which in this case is null
    $this->assertNull($mock->foo());
}

public function testResettingCallRecorder()
{
    $mock = Phake::mock(PhakeTest_MockedClass::class);
    $mock->foo();

    //Will work as normal
    Phake::verify($mock)->foo();

    Phake::reset($mock);

    //Will now throw an error that foo was not called
    Phake::verify($mock)->foo();
}

Stubbing Multiple Calls

Another benefit of the stub mapping in Phake is that it allows you to very easily stub multiple calls to the same method that use different parameters. In my shopping cart I have decided to add some functionality that will allow me to easily add multiple products that are a part of a group to the shopping cart. To facilitate this I have decided to create a new class called ItemGroup. The ItemGroup object will be constructed with an array of Items. It will have a method on the class that will add all of the items in the group to the given cart and then the total price of items in the cart will be returned.

It should be noted that earlier I decided to make a small change to the ShoppingCart::addItem() method to have it return the total price of items in the cart. I figured that this would be nice api level functionality to make working with the system a little bit easier. I would like to take advantage of that change with this code. Here's a stub of the functionality I am considering.

/**
 * A group of items that can be added to a cart all at the same time
 */
class ItemGroup
{
    /**
     * @param array $items an array of Item objects
     */
    public function __construct(array $items)
    {
    }

    /**
     * @param ShoppingCart $cart
     * @return money The new total value of the cart
     */
    public function addItemsToCart(ShoppingCart $cart)
    {
    }
}

The next test I am going to write now is going to be focusing on this new ItemGroup::addItemsToCart() method. In my test's setUp() method I'll create a new instance of ItemGroup which will require one or more Item implementations. I'll use mocks for those. Then the actual test case I am going to start with will be a test to assert that ItemGroup::addItemsToCart() returns the new shopping cart value. I already know that I am going to need to get this value by looking at the last return value from calls to ShoppingCart::addItem(). To allow for checking this I will mock ShoppingCart and create three stubs for ShoppingCart::addItem(). Each stub will be for a call with a different Item.

class ItemGroupTest extends PHPUnit\Framework\TestCase
{
    private ItemGroup $itemGroup;
    private Item $item1;
    private Item $item2;
    private Item $item3;

    public function setUp(): void
    {
        $this->item1 = Phake::mock(Item::class);
        $this->item2 = Phake::mock(Item::class);
        $this->item3 = Phake::mock(Item::class);

        $this->itemGroup = new ItemGroup([ $this->item1, $this->item2, $this->item3 ]);
    }

    public function testAddItemsToCart()
    {
        $cart = Phake::mock(ShoppingCart::class);
        Phake::when($cart)->addItem($this->item1)->thenReturn(10);
        Phake::when($cart)->addItem($this->item2)->thenReturn(20);
        Phake::when($cart)->addItem($this->item3)->thenReturn(30);

        $totalCost = $this->itemGroup->addItemsToCart($cart);
        $this->assertEquals(30, $totalCost);
    }
}

In this example the ShoppingCart::addItem() method is being stubbed three times. Each time it is being stubbed with a different parameter being passed to addItem(). This a good example of how parameters are also checked whenever Phake looks at a mock object's stub map for answers. The default behavior of argument matching is again a loose equality check. Similar to how you would use the double equals operator in PHP. The other options for argument matching are discussed further in Method Parameter Matchers.

Stubbing Consecutive Calls

The previous test was a great example for how you can make multiple stubs for a single method however in reality it is not the best way for that particular test to be written. What if the Item objects in an ItemGroup aren't stored in the order they were passed in? I am needlessly binding my test to the order in which objects are stored. Phake provides the ability to map multiple answers to the same stub. This is done simply by chaining the answers together. I could rewrite the test from the previous chapter to utilize this feature of Phake.

class ItemGroupTest extends PHPUnit\Framework\TestCase
{
    private ItemGroup $itemGroup;
    private Item $item1;
    private Item $item2;
    private Item $item3;

    public function setUp(): void
    {
        $this->item1 = Phake::mock(Item::class);
        $this->item2 = Phake::mock(Item::class);
        $this->item3 = Phake::mock(Item::class);

        $this->itemGroup = new ItemGroup([ $this->item1, $this->item2, $this->item3 ]);
    }

    public function testAddItemsToCart()
    {
        $cart = Phake::mock(ShoppingCart::class);
        Phake::when($cart)->addItem(Phake::anyParameters())->thenReturn(10)
            ->thenReturn(20)
            ->thenReturn(30);

        $totalCost = $this->itemGroup->addItemsToCart($cart);
        $this->assertEquals(30, $totalCost);
    }
}

You will notice a few of differences between this example and the example in Stubbing Multiple Calls. The first difference is that there is only one call to Phake::when(). The second difference is that I have chained together three calls to thenReturn(). The third difference is instead of passing one of my mock Item objects I have passed the result of the Phake::anyParameters() method. This is a special argument matcher in Phake that essentially says match any call to the method regardless of the number of parameters or the value of those parameters. You can learn more about Phake::anyParameters() in Wildcard Parameters.

So, this single call to Phake::when() is saying: "Whenever a call to $cart->addItem() is made, regardless of the parameters, return 10 for the first call, 20 for the second call, and 30 for the third call." If you are using consecutive call stubbing and you call the method more times than you have answers set, the last answer will continue to be returned. In this example, if $cart->addItem() were called a fourth time, then 30 would be returned again.

Stubbing Reference Parameters

Occasionally you may run into code that utilizes reference parameters to provide additional output from a method. This is not an uncommon thing to run into with legacy code. Phake provides a custom parameter matcher (these are discussed further in Method Parameter Matchers) that allows you to set reference parameters. It can be accessed using Phake::setReference(). The only parameter to this matcher is the value you would like to set the reference parameter to provided all other parameters match.

interface IValidator
{
    /**
     * @parm array $data Data to validate
     * @parm array &$errors contains all validation errors if the data is not valid
     * @return boolean True when the data is valid
     */
    public function validate(array $data, array &$errors);
}

class ValidationLogger implements IValidator
{
    private $validator;
    private $log;

    public function __construct(IValidator $validator, Logger $log)
    {
        $this->validator = $validator;
        $this->log = $log;
    }

    public function validate(array $data, array &$errors)
    {
        if (!$this->validator->validate($data, $errors))
        {
            foreach ($errors as $error)
            {
                $this->log->info("Validation Error: {$error}");
            }

            return false;
        }

        return true;
    }
}

class ValidationLoggerTest extends PHPUnit\Framework\TestCase
{
    public function testValidate()
    {
        //Mock the dependencies
        $validator = Phake::mock(IValidator::class);
        $log = Phake::mock(Logger::class);
        $data = [ 'data1' => 'value' ];
        $expectedErrors = ['data1 is not valid'];

        //Setup the stubs (Notice the Phake::setReference()
        Phake::when($validator)->validate($data, Phake::setReference($expectedErrors))->thenReturn(false);

        //Instantiate the SUT
        $validationLogger = new ValidationLogger($validator, $log);

        //verify the validation is false and the message is logged
        $errors = [];
        $this->assertFalse($validationLogger->validate($data, $errors));
        Phake::verify($log)->info('Validation Error: data1 is not valid');
    }
}

In the example above, I am testing a new class I have created called ValidationLogger. It is a decorator for other implementations of IValidator that allows adding logging to any other validator. The IValidator::validate() method will always return an array of errors into the second parameter (a reference parameter) provided to the method. These errors are what my logger is responsible for logging. So in order for my test to work properly, I will need to be able to set that second parameter as a part of my stubbing call.

In the call to Phake::when($validator)->validate() I have passed a call to Phake::setReference() as the second parameter. This is causing the mock implementation of IValidator to set $errors in ValidationLogger::validate() to the array specified by $expectedErrors. This allows me to quickly and easily validate that I am actually logging the errors returned back in the reference parameter.

By default Phake::setReference() will always return true regardless of the parameter initially passed in. If you would like to only set a reference parameter when that reference parameter was passed in as a certain value you can use the when() modifier. This takes a single parameter matcher as an argument. Below, you will see that the test has been modified to call when() on the result of Phake::setReference(). This modification will cause the reference parameter to be set only if the $errors parameter passed to IValidator::validate() is initially passed as an empty array.

class ValidationLoggerTest extends PHPUnit\Framework\TestCase
{
    public function testValidate()
    {
        //Mock the dependencies
        $validator = Phake::mock(IValidator::class);
        $log = Phake::mock(Logger::class);
        $data = [ 'data1' => 'value' ];
        $expectedErrors = [ 'data1 is not valid' ];

        //Setup the stubs (Notice the Phake::setReference()
        Phake::when($validator)->validate($data, Phake::setReference($expectedErrors)->when([])->thenReturn(false);

        //Instantiate the SUT
        $validationLogger = new ValidationLogger($validator, $log);

        //verify the validation is false and the message is logged
        $errors = [];
        $this->assertFalse($validationLogger->validate($data, $errors));
        Phake::verify($log)->info('Validation Error: data1 is not valid');
    }
}

Please note, when you are using Phake::setReference() you still must provide an answer for the stub. If you use this function and your reference parameter is never changed, that is generally the most common reason.

Partial Mocks

When testing legacy code, if you find that the majority of the methods in the mock are using the thenCallParent() answer, you may find it easier to just use a partial mock in Phake. Phake partial mocks also allow you to call the actual constructor of the class being mocked. They are created using Phake::partialMock(). Like Phake::mock(), the first parameter is the name of the class that you are mocking. However, you can pass additional parameters that will then be passed as the respective parameters to that class’ constructor. The other notable feature of a partial mock in Phake is that its default answer is to pass the call through to the parent as if you were using thenCallParent().

Consider the following class that has a method that simply returns the value passed into the constructor.

class MyClass
{
    private $value;

    public __construct($value)
    {
        $this->value = $value;
    }

    public function foo()
    {
        return $this->value;
    }
}

Using Phake::partialMock() you can instantiate a mock object that will allow this object to function as designed while still allowing verification as well as selective stubbing of certain calls. Below is an example that shows the usage of Phake::partialMock().

class MyClassTest extends PHPUnit\Framework\TestCase
{
    public function testCallingParent()
    {
        $mock = Phake::partialMock(MyClass::class, 42);

        $this->assertEquals(42, $mock->foo());
    }
}

Again, partial mocks should not be used when you are testing new code. If you find yourself using them be sure to inspect your design to make sure that the class you are creating a partial mock for is not doing too much.

Setting Default Stubs

You can also change the default stubbing for mocks created with Phake::mock(). This is done by using the second parameter to Phake::mock() in conjunction with the Phake::ifUnstubbed() method. The second parameter to Phake::mock() is reserved for configuring the behavior of an individual mock. Phake::ifUnstubbed() allows you to specify any of the matchers mentioned above as the default answer if any method invocation is not explicitly stubbed. If this configuration directive is not provided then the method will return NULL by default. An example of this can be seen below.

class MyClassTest extends PHPUnit\Framework\TestCase
{
    public function testDefaultStubs()
    {
        $mock = Phake::mock(MyClass::class, Phake::ifUnstubbed()->thenReturn(42));

        $this->assertEquals(42, $mock->foo());
    }
}

Stubbing Magic Methods

Most magic methods can be stubbed using the method name just like you would any other method. The one exception to this is the __call() method. This method is overwritten on each mock already to allow for the fluent api that Phake utilizes. If you want to stub a particular invocation of __call() you can create a stub for the method you are targetting in the first parameter to __call().

Consider the following class.

class MagicClass
{
    public function __call($method, $args)
    {
        return '__call';
    }
}

You could stub an invocation of the __call() method through a userspace call to magicCall() with the following code.

class MagicClassTest extends PHPUnit\Framework\TestCase
{
    public function testMagicCall()
    {
        $mock = Phake::mock(MagicClass::class);

        Phake::when($mock)->magicCall()->thenReturn(42);

        $this->assertEquals(42, $mock->magicCall());
    }
}

If for any reason you need to explicitly stub calls to __call() then you can use Phake::whenCallMethodWith(). The matchers passed to Phake::whenCallMethod() will be matched to the method name and array of arguments similar to what you would expect to be passed to a __call() method. You can also use Phake::anyParameters() instead.

class MagicClassTest extends PHPUnit\Framework\TestCase
{
    public function testMagicCall()
    {
        $mock = Phake::mock(MagicClass::class);

        Phake::whenCallMethodWith('magicCall', [])->isCalledOn($mock)->thenReturn(42);

        $this->assertEquals(42, $mock->magicCall());
    }
}