stratadox/clock

v0.2 2021-03-18 09:10 UTC

README

Factory for creating datetime objects.

Build Status Coverage Status Scrutinizer Code Quality

What

It's really as simple as the name suggests: this is a clock, used to indicate what time it is.

Since it produces DateTime objects, this clock is somewhat special in the sense that it can also read the date.

Just a clock, really.

Why

  • As soon as you use "unadulterated" datetime objects in your code, any test you've written for it immediately risks being flaky, because if there's a tiny bit of time between new DateTime and your assertion, the test fails.
  • Instantiating a new DateTime or new DateTimeImmutable in client code, is a static invocation. This introduces coupling and reduces testability. This obviously goes double for date_create() and the like.
  • It's a lot more natural to get the time from a clock than to instantiate a new instant each time you want to know how late it is.

Installing

Install with composer require stratadox/clock

Examples

Clock example

In a service that needs to know the time:

<?php
namespace Your\Project;

use Stratadox\Clock\Clock;

class SomeFactory
{
    private $clock;

    public function __construct(Clock $clock)
    {
        $this->clock = $clock;
    }

    public function createSomething(): Something
    {
        return new Something($this->clock->now());
    }
}

class Something
{
    private $creationDate;

    public function __construct(\DateTimeInterface $creationDate)
    {
        $this->creationDate = $creationDate;
    }

    public function creationDate(): \DateTimeInterface
    {
        return $this->creationDate;
    }
}

Rewindable clock example

In a service that needs to rewind or fast-forward the clock:

<?php
namespace Your\Project;

use Stratadox\Clock\RewindableClock;

class Scheduler
{
    private $clock;

    public function __construct(RewindableClock $clock)
    {
        $this->clock = $clock;
    }

    public function scheduleForTheNextThreeHours(): Schedule
    {
        return new Schedule(
            new Activity($this->clock->now()),
            new Activity($this->clock->fastForward(new \DateInterval('PT1H'))->now()),
            new Activity($this->clock->fastForward(new \DateInterval('PT2H'))->now())
        );
    }

    public function whenDoIWantThisOnMyDesk(): \DateTimeInterface
    {
        return $this->clock->rewind(new \DateInterval('P1D'))->now();
    }
}

Choosing a clock

The default implementation is the DateTimeClock. It produces a new DateTimeImmutable object whenever now is called.

If the datetime object needs to be passed into something that has a DateTime type hint, or otherwise relies on mutable datetime objects, it is preferable to solve that issue. For example by replacing the DateTime hint with DateTimeIterface or DateTimeImmutable, or passing along a RewindableClock. In cases where that is not an option, the DateTimeMutableClock clock can be used instead.

In case timezones are important in your context, there is also a TimeZoneAwareClock, which takes a timezone as constructor parameter.

To prevent the clock from ticking while other code is running, your tests can instantiate and inject an UnmovingClock.

In a service definition:

<?php
use Stratadox\Clock\Clock;
use Stratadox\Clock\DateTimeClock;

$container->set(Clock::class, function () {
    return DateTimeClock::create();
});

In a unit test:

<?php
use Your\Project\SomeFactory;
use PHPUnit\Framework\TestCase;
use Stratadox\Clock\UnmovingClock;

class SomethingTest extends TestCase
{
    public function testCreatingSomething(): void
    {
        $testTime = new DateTimeImmutable('1-1-1960');
        $factory = new SomeFactory(
            UnmovingClock::standingStillAt($testTime)
        );

        $something = $factory->createSomething();

        $this->assertEquals($testTime, $something->creationDate());
    }
}

When the clock needs to be able to rewind or fast-forward, use the RewindableDateTimeClock implementation.

In a service definition:

<?php
use Stratadox\Clock\RewindableClock;
use Stratadox\Clock\RewindableDateTimeClock;

$container->set(RewindableClock::class, function () {
    return RewindableDateTimeClock::create();
});

In a unit test:

<?php
use Your\Project\Scheduler;
use PHPUnit\Framework\TestCase;
use Stratadox\Clock\RewindableDateTimeClock;
use Stratadox\Clock\UnmovingClock;

class SchedulerTest extends TestCase
{
    public function testWhenItShouldBeOnTheirDesk(): void
    {
        $scheduler = new Scheduler(
            RewindableDateTimeClock::using(UnmovingClock::standingStillAt(
                new DateTimeImmutable('16-5-1991')
            ))
        );

        $this->assertEquals(
            new DateTimeImmutable('15-5-1991'),
            $scheduler->whenDoIWantThisOnMyDesk()
        );
    }
}

Here's an overview of which clock to use when:

Clock When to use
DateTimeClock In most situations
DateTimeMutableClock When you can't use DateTimeImmutable
TimeZoneAwareClock When the default timezone isn't enough
TimeZoneAwareMutableClock When both two previous reasons apply
UnmovingClock During testing
RewindableDateTimeClock If the clock needs to be set back or forth

Note that the rewindable clock can be combined with any other clocks, in order to produce, for instance, rewinded mutable datetime objects in a particular timezone.