From b09f92223a356567d2e6688b5993cc58a2c33f46 Mon Sep 17 00:00:00 2001
From: Nathaniel Catchpole <catch@35733.no-reply.drupal.org>
Date: Thu, 13 Oct 2016 10:42:53 +0100
Subject: [PATCH] Issue #2717207 by mpdonadio, dawehner, catch, alexpott: Add a
 time service to provide consistent interaction with time()/microtime() and
 superglobals

---
 core/core.services.yml                        |   3 +
 core/includes/bootstrap.inc                   |   3 +
 core/lib/Drupal.php                           |  10 ++
 core/lib/Drupal/Component/Datetime/Time.php   |  57 +++++++
 .../Component/Datetime/TimeInterface.php      | 149 ++++++++++++++++++
 .../Tests/Component/Datetime/TimeTest.php     | 123 +++++++++++++++
 6 files changed, 345 insertions(+)
 create mode 100644 core/lib/Drupal/Component/Datetime/Time.php
 create mode 100644 core/lib/Drupal/Component/Datetime/TimeInterface.php
 create mode 100644 core/tests/Drupal/Tests/Component/Datetime/TimeTest.php

diff --git a/core/core.services.yml b/core/core.services.yml
index d1223e9cc757..cb935793a6b0 100644
--- a/core/core.services.yml
+++ b/core/core.services.yml
@@ -346,6 +346,9 @@ services:
     class: Drupal\Core\Database\Connection
     factory: Drupal\Core\Database\Database::getConnection
     arguments: [default]
+  datetime.time:
+    class: Drupal\Component\Datetime\Time
+    arguments: ['@request_stack']
   file_system:
     class: Drupal\Core\File\FileSystem
     arguments: ['@stream_wrapper_manager', '@settings', '@logger.channel.file']
diff --git a/core/includes/bootstrap.inc b/core/includes/bootstrap.inc
index 71895b2da37a..f74859e17185 100644
--- a/core/includes/bootstrap.inc
+++ b/core/includes/bootstrap.inc
@@ -84,6 +84,9 @@
  *
  * @see http://php.net/manual/reserved.variables.server.php
  * @see http://php.net/manual/function.time.php
+ *
+ * @deprecated in Drupal 8.3.0, will be removed before Drupal 9.0.0.
+ *   Use \Drupal::time()->getRequestTime();
  */
 define('REQUEST_TIME', (int) $_SERVER['REQUEST_TIME']);
 
diff --git a/core/lib/Drupal.php b/core/lib/Drupal.php
index bc21709b5095..81c72db081aa 100644
--- a/core/lib/Drupal.php
+++ b/core/lib/Drupal.php
@@ -738,4 +738,14 @@ public static function entityDefinitionUpdateManager() {
     return static::getContainer()->get('entity.definition_update_manager');
   }
 
+  /**
+   * Returns the time service.
+   *
+   * @return \Drupal\Component\Datetime\TimeInterface
+   *   The time service.
+   */
+  public static function time() {
+    return static::getContainer()->get('datetime.time');
+  }
+
 }
diff --git a/core/lib/Drupal/Component/Datetime/Time.php b/core/lib/Drupal/Component/Datetime/Time.php
new file mode 100644
index 000000000000..cc6f2a29dbd9
--- /dev/null
+++ b/core/lib/Drupal/Component/Datetime/Time.php
@@ -0,0 +1,57 @@
+<?php
+
+namespace Drupal\Component\Datetime;
+
+use Symfony\Component\HttpFoundation\RequestStack;
+
+/**
+ * Provides a class for obtaining system time.
+ */
+class Time implements TimeInterface {
+
+  /**
+   * The request stack.
+   *
+   * @var \Symfony\Component\HttpFoundation\RequestStack
+   */
+  protected $requestStack;
+
+  /**
+   * Constructs a Time object.
+   *
+   * @param \Symfony\Component\HttpFoundation\RequestStack $request_stack
+   *   The request stack.
+   */
+  public function __construct(RequestStack $request_stack) {
+    $this->requestStack = $request_stack;
+  }
+
+  /**
+   * @{inheritdoc}
+   */
+  public function getRequestTime() {
+    return $this->requestStack->getCurrentRequest()->server->get('REQUEST_TIME');
+  }
+
+  /**
+   * @{inheritdoc}
+   */
+  public function getRequestMicroTime() {
+    return $this->requestStack->getCurrentRequest()->server->get('REQUEST_TIME_FLOAT');
+  }
+
+  /**
+   * @{inheritdoc}
+   */
+  public function getCurrentTime() {
+    return time();
+  }
+
+  /**
+   * @{inheritdoc}
+   */
+  public function getCurrentMicroTime() {
+    return microtime(TRUE);
+  }
+
+}
diff --git a/core/lib/Drupal/Component/Datetime/TimeInterface.php b/core/lib/Drupal/Component/Datetime/TimeInterface.php
new file mode 100644
index 000000000000..0c81575c73a1
--- /dev/null
+++ b/core/lib/Drupal/Component/Datetime/TimeInterface.php
@@ -0,0 +1,149 @@
+<?php
+
+namespace Drupal\Component\Datetime;
+
+/**
+ * Defines an interface for obtaining system time.
+ */
+interface TimeInterface {
+
+  /**
+   * Returns the timestamp for the current request.
+   *
+   * This method should be used to obtain the current system time at the start
+   * of the request. It will be the same value for the life of the request
+   * (even for long execution times).
+   *
+   * This method can replace instances of
+   * @code
+   * $request_time = $_SERVER['REQUEST_TIME'];
+   * $request_time = REQUEST_TIME;
+   * $request_time = $requestStack->getCurrentRequest()->server->get('REQUEST_TIME');
+   * $request_time = $request->server->get('REQUEST_TIME');
+   * @endcode
+   * and most instances of
+   * @code
+   * $time = time();
+   * @endcode
+   * with
+   * @code
+   * $request_time = \Drupal::time()->getRequestTime();
+   * @endcode
+   * or the equivalent using the injected service.
+   *
+   * Using the time service, rather than other methods, is especially important
+   * when creating tests, which require predictable timestamps.
+   *
+   * @return int
+   *   A Unix timestamp.
+   *
+   * @see \Drupal\Component\Datetime\TimeInterface::getRequestMicroTime()
+   * @see \Drupal\Component\Datetime\TimeInterface::getCurrentTime()
+   * @see \Drupal\Component\Datetime\TimeInterface::getCurrentMicroTime()
+   */
+  public function getRequestTime();
+
+  /**
+   * Returns the timestamp for the current request with microsecond precision.
+   *
+   * This method should be used to obtain the current system time, with
+   * microsecond precision, at the start of the request. It will be the same
+   * value for the life of the request (even for long execution times).
+   *
+   * This method can replace instances of
+   * @code
+   * $request_time_float = $_SERVER['REQUEST_TIME_FLOAT'];
+   * $request_time_float = $requestStack->getCurrentRequest()->server->get('REQUEST_TIME_FLOAT');
+   * $request_time_float = $request->server->get('REQUEST_TIME_FLOAT');
+   * @endcode
+   * and many instances of
+   * @code
+   * $microtime = microtime();
+   * $microtime = microtime(TRUE);
+   * @endcode
+   * with
+   * @code
+   * $request_time = \Drupal::time()->getRequestMicroTime();
+   * @endcode
+   * or the equivalent using the injected service.
+   *
+   * Using the time service, rather than other methods, is especially important
+   * when creating tests, which require predictable timestamps.
+   *
+   * @return float
+   *   A Unix timestamp with a fractional portion.
+   *
+   * @see \Drupal\Component\Datetime\TimeInterface::getRequestTime()
+   * @see \Drupal\Component\Datetime\TimeInterface::getCurrentTime()
+   * @see \Drupal\Component\Datetime\TimeInterface::getCurrentMicroTime()
+   */
+  public function getRequestMicroTime();
+
+  /**
+   * Returns the current system time as an integer.
+   *
+   * This method should be used to obtain the current system time, at the time
+   * the method was called.
+   *
+   * This method can replace many instances of
+   * @code
+   * $time = time();
+   * @endcode
+   * with
+   * @code
+   * $request_time = \Drupal::time()->getCurrentTime();
+   * @endcode
+   * or the equivalent using the injected service.
+   *
+   * This method should only be used when the current system time is actually
+   * needed, such as with timers or time interval calculations. If only the
+   * time at the start of the request is needed,
+   * use TimeInterface::getRequestTime().
+   *
+   * Using the time service, rather than other methods, is especially important
+   * when creating tests, which require predictable timestamps.
+   *
+   * @return int
+   *   A Unix timestamp.
+   *
+   * @see \Drupal\Component\Datetime\TimeInterface::getRequestTime()
+   * @see \Drupal\Component\Datetime\TimeInterface::getRequestMicroTime()
+   * @see \Drupal\Component\Datetime\TimeInterface::getCurrentMicroTime()
+   */
+  public function getCurrentTime();
+
+  /**
+   * Returns the current system time with microsecond precision.
+   *
+   * This method should be used to obtain the current system time, with
+   * microsecond precision, at the time the method was called.
+   *
+   * This method can replace many instances of
+   * @code
+   * $microtime = microtime();
+   * $microtime = microtime(TRUE);
+   * @endcode
+   * with
+   * @code
+   * $request_time = \Drupal::time()->getCurrentMicroTime();
+   * @endcode
+   * or the equivalent using the injected service.
+   *
+   * This method should only be used when the current system time is actually
+   * needed, such as with timers or time interval calculations. If only the
+   * time at the start of the request and microsecond precision is needed,
+   * use TimeInterface::getRequestMicroTime().
+   *
+   * Using the time service, rather than other methods, is especially important
+   * when creating tests, which require predictable timestamps.
+   *
+   * @return float
+   *   A Unix timestamp with a fractional portion.
+   *
+   * @see \Drupal\Component\Datetime\TimeInterface::getRequestTime()
+   * @see \Drupal\Component\Datetime\TimeInterface::getRequestMicroTime()
+   * @see \Drupal\Component\Datetime\TimeInterface::getCurrentTime()
+   */
+  public function getCurrentMicroTime();
+
+}
diff --git a/core/tests/Drupal/Tests/Component/Datetime/TimeTest.php b/core/tests/Drupal/Tests/Component/Datetime/TimeTest.php
new file mode 100644
index 000000000000..0553451de8c4
--- /dev/null
+++ b/core/tests/Drupal/Tests/Component/Datetime/TimeTest.php
@@ -0,0 +1,123 @@
+<?php
+
+namespace Drupal\Tests\Component\Datetime;
+
+use Drupal\Component\Datetime\Time;
+use Drupal\Tests\UnitTestCase;
+use Symfony\Component\HttpFoundation\Request;
+
+/**
+ * @coversDefaultClass \Drupal\Component\Datetime\Time
+ * @group Datetime
+ *
+ * Isolate the tests to prevent side effects from altering system time.
+ *
+ * @runTestsInSeparateProcesses
+ * @preserveGlobalState disabled
+ */
+class TimeTest extends UnitTestCase {
+
+  /**
+   * The mocked request stack.
+   *
+   * @var \Symfony\Component\HttpFoundation\RequestStack|\PHPUnit_Framework_MockObject_MockObject
+   */
+  protected $requestStack;
+
+  /**
+   * The mocked time class.
+   *
+   * @var \Drupal\Component\Datetime\Time
+   */
+  protected $time;
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp() {
+    parent::setUp();
+
+    $this->requestStack = $this->getMock('Symfony\Component\HttpFoundation\RequestStack');
+
+    $this->time = new Time($this->requestStack);
+  }
+
+  /**
+   * Tests the getRequestTime method.
+   *
+   * @covers ::getRequestTime
+   */
+  public function testGetRequestTime() {
+    $expected = 12345678;
+
+    $request = Request::createFromGlobals();
+    $request->server->set('REQUEST_TIME', $expected);
+
+    // Mocks a the request stack getting the current request.
+    $this->requestStack->expects($this->any())
+      ->method('getCurrentRequest')
+      ->willReturn($request);
+
+    $this->assertEquals($expected, $this->time->getRequestTime());
+  }
+
+  /**
+   * Tests the getRequestMicroTime method.
+   *
+   * @covers ::getRequestMicroTime
+   */
+  public function testGetRequestMicroTime() {
+    $expected = 1234567.89;
+
+    $request = Request::createFromGlobals();
+    $request->server->set('REQUEST_TIME_FLOAT', $expected);
+
+    // Mocks a the request stack getting the current request.
+    $this->requestStack->expects($this->any())
+      ->method('getCurrentRequest')
+      ->willReturn($request);
+
+    $this->assertEquals($expected, $this->time->getRequestMicroTime());
+  }
+
+  /**
+   * Tests the getCurrentTime method.
+   *
+   * @covers ::getCurrentTime
+   */
+  public function testGetCurrentTime() {
+    $expected = 12345678;
+    $this->assertEquals($expected, $this->time->getCurrentTime());
+  }
+
+  /**
+   * Tests the getCurrentMicroTime method.
+   *
+   * @covers ::getCurrentMicroTime
+   */
+  public function testGetCurrentMicroTime() {
+    $expected = 1234567.89;
+    $this->assertEquals($expected, $this->time->getCurrentMicroTime());
+  }
+
+}
+
+namespace Drupal\Component\Datetime;
+
+/**
+ * Shadow time() system call.
+ *
+ * @returns int
+ */
+function time() {
+  return 12345678;
+}
+
+/**
+ * Shadow microtime system call.
+ *
+ * @returns float
+ */
+function microtime() {
+  return 1234567.89;
+}
-- 
GitLab