A common problem in Unit Testing in PHP is testing something that depends on the current time. For a determined test it should be possible to set the time in your test script without really changing the system settings. In this article I’ll describe how it is usually done with OOP and then come to an alternative solution with much less code that makes use of the new features in PHP 5.3.
The usual approach would be a wrapper class like this:
class Calendar { public function time() { return time(); } public function date($format, $time = null) { return date($format, $time ?: $this->time()); } // ... }
Now any class that uses date/time functions has to be modified to use the Calendar class via Dependency Injection:
class SomeClass { /** * @var Calendar */ private $calendar; public function __construct(Calendar $calendar = null) { $this->calendar = $calendar ?: new Calendar; } public function oneHourAgo() { return $this->calendar->date('H:i:s', $this->calendar->time() - 3600); } }
Then you mock the Calendar class in your tests and pass it to the test subject. I won’t go into further details because you probaly know the concept of mocking and how to do this in your favourite unit testing framework. After all this article is not about mocking classes, because I have:
A simpler solution with namespaces
If you are using PHP 5.3 namespaces you are lucky because you won’t need all this overhead and probably no changes in your classes at all. The trick is to override built-in functions in your current namespace. Consider this namespaced version of the class from above:
namespace My\Namespace; class SomeClass { public function oneHourAgo() { return date('H:i:s', time() - 3600); } }
As you can see, no overhead, just a straightforward call to date() and time(). To test this with specific times we implement a test case as follows (Example in PHPUnit but works as well with other frameworks):
namespace My\Namespace; require_once 'PHPUnit\Framework\TestCase.php'; /** * Override time() in current namespace for testing * * @return int */ function time() { return SomeClassTest::$now ?: \time(); } class SomeClassTest extends \PHPUnit_Framework_TestCase { /** * @var int $now Timestamp that will be returned by time() */ public static $now; /** * @var SomeClass $someClass Test subject */ private $someClass; /** * Create test subject before test */ protected function setUp() { parent::setUp(); $this->someClass = new SomeClass; } /** * Reset custom time after test */ protected function tearDown() { self::$now = null; } /* * Test cases */ public function testOneHourAgoFromNoon() { self::$now = strtotime('12:00'); $this->assertEquals('11:00', $this->someClass->oneHourAgo()); } public function testOneHourAgoFromMidnight() { self::$now = strtotime('0:00'); $this->assertEquals('23:00', $this->someClass->oneHourAgo()); } }
The crucial point here is that we implement a new function named exaclty like a built-in function. You cannot replace functions but since this is defined in the namespace \My\Namespace
it does not replace anything. In fact it is a new function with the fully qualified name \My\Namespace\time()
The test subject now calls time()
as unqualified name so PHP looks for the function in the current namespace at first. That is \My\Namespace\time()
in our example. I recommend the section about name resolution rules in the manual for further reading.
Important Implication: It does not work if you use the global functions with fully qualified names (i.E. \time()
) in your test subjects!
You can implement this function however you like, I decided to make the return value configurable within the test case via a static property that gets resetted after each test and if it is not set the real time is used.
I hope this solution helps, it may feel hackish but for me it made testing a lot easier!
This looks excellent. The cleanest way to mock time that I have seen so far !
Thanks a lot for the tip ๐
I liked this strategy so much… I pushed it a little further down the line ๐
In this post in programmers stackexchange I was searching for a way to organize my tests.
I finally implemented a solution following the steps you describe in this page, overriding the DateTime() function. Available here for the curious ๐
Thanks again !
I recently implemented the library php-mock which uses that language feature for mocking non deterministic PHP functions like time().
Very nice library, good job! Definitely going to use that when I need it the next time!
In time() function better return ?: \time(); otherwise it will just call itself
Thanks for noticing! I updated the code.
I’ve just refactored this method for my CakePHP 3 app in it works a treat. Thank you so much!
“Important Implication: It does not work if you use the global functions with fully qualified names (i.E. \time()) in your test subjects!”
So, any ideas on how to test when the fully qualified name is used?
With the runkit extension, you can redefine PHP functions. I don’t know any better method, except refactoring the code under test. See also: http://stackoverflow.com/questions/2371854/can-i-mock-time-in-phpunit
Huge help! Made it easy to essentially “disable” the mail() function in our Email class by simply returning true when the mail() function is called.
The problem with this approach is it requires you to make assumptions about the internal workings of a given class, meaning that if you change how a class actually works then your unit tests might break even though the class itself is still fine.