Recently, while starting to learn Ruby, I’ve stumbled upon the following code snippet which demonstrates how you can assert that a piece of code throws as exception in Ruby’s Test::Unit:
1 2 3 4 5 6 7 8 |
|
Check out lines 5-7 – isn’t this a really elegant and readable way of expecting an exception? Now let’s contrast this with how it’s done in PHPUnit:
1 2 3 4 5 6 7 8 9 10 |
|
Well, that isn’t too horrible but I’ve never been really happy with this.
My main issues with this way of expecting exceptions are:
The expectation is pretty far away from the location you’d normally expect to find an assertion. Usually, an assertion can be found at the bottom of each test function, whereas with the current method PHPUnit uses, it’s at the top of the test-function.
Additionally, it’s an annotation “buried” in a comment which is easy to miss.
Finally, PHPUnit will watch for an exception thrown by any of the code inside the test-function. Normally, it’s the last line of the test-function, so it isn’t hard to find – but what if the expected exception is thrown in a line before the last line, maybe due to a bug?
So, is there a way to mimic the method of Test::Unit in PHPUnit?
As you may have guessed, there is one, using anonymous functions and closures, which are available since PHP 5.3!
Here’s the code for a simple function we can use to check if a given piece of code throws an exception:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
Confused?
Well, the most important line of that function is line 5.
There, the anonymous function that was passed into the function as the second argument gets executed, which allows
us to catch any exception thrown by it.
Still confused? Let’s check out an example:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
The magic happens in lines 10-13:
The execute-method of ExceptionThrower doesn’t get executed immediately because it’s wrapped in an anonymous function, i.e.
execute will only be invoked when we call the anonymous function contained in $code in line 5 of assertThrowsException().
Now one question remains – can this be done in PHPUnit with a “proper” assertion?
Yes, it can!
I won’t post the full code into this blog, it would be a bit too much but you can find the code adding a new assertion
assertThrowsException() in the following commit to my fork of PHPUnit 3.5:
Link to Github commit
Note that this code is not production-ready, it’s a proof-of-concept with just enough code to get PHPUnit to run the
new assertion without throwing a fit, just so I could find out whether such a new assertion would be possible at all.
Anyway, with the new assertion the PHPUnit test-method from the beginning of this blog would look like this:
1 2 3 4 5 6 7 8 9 |
|
Granted, you have more to type than in the current way but in my opinion, this offers more control and is more expressive and I’d prefer expressiveness over terseness any day.
What do you think, should I turn this into a real patch for PHPUnit?
Or am I just crazy? ;)