3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2 of the License, or
6 * (at your option) any later version.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License along
14 * with this program; if not, write to the Free Software Foundation, Inc.,
15 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
16 * http://www.gnu.org/copyleft/gpl.html
21 use Psr\Log\LoggerAwareInterface
;
22 use Psr\Log\LoggerInterface
;
23 use Psr\Log\NullLogger
;
26 * An interface to help developers measure the performance of their applications.
27 * This interface closely matches the W3C's User Timing specification.
28 * The key differences are:
30 * - The reference point for all measurements which do not explicitly specify
31 * a start time is $_SERVER['REQUEST_TIME_FLOAT'], not navigationStart.
32 * - Successive calls to mark() and measure() with the same entry name cause
33 * the previous entry to be overwritten. This ensures that there is a 1:1
34 * mapping between names and entries.
35 * - Because there is a 1:1 mapping, instead of getEntriesByName(), we have
38 * The in-line documentation incorporates content from the User Timing Specification
39 * https://www.w3.org/TR/user-timing/
40 * Copyright © 2013 World Wide Web Consortium, (MIT, ERCIM, Keio, Beihang).
41 * https://www.w3.org/Consortium/Legal/2015/doc-license
45 class Timing
implements LoggerAwareInterface
{
48 private $entries = [];
50 /** @var LoggerInterface */
53 public function __construct( array $params = [] ) {
55 $this->setLogger( $params['logger'] ??
new NullLogger() );
59 * Sets a logger instance on the object.
61 * @param LoggerInterface $logger
63 public function setLogger( LoggerInterface
$logger ) {
64 $this->logger
= $logger;
68 * Store a timestamp with the associated name (a "mark")
70 * @param string $markName The name associated with the timestamp.
71 * If there already exists an entry by that name, it is overwritten.
72 * @return array The mark that has been created.
74 public function mark( $markName ) {
75 $this->entries
[$markName] = [
77 'entryType' => 'mark',
78 'startTime' => microtime( true ),
81 return $this->entries
[$markName];
85 * @param string|null $markName The name of the mark that should
86 * be cleared. If not specified, all marks will be cleared.
88 public function clearMarks( $markName = null ) {
89 if ( $markName !== null ) {
90 unset( $this->entries
[$markName] );
94 'name' => 'requestStart',
95 'entryType' => 'mark',
96 'startTime' => $_SERVER['REQUEST_TIME_FLOAT'],
104 * This method stores the duration between two marks along with
105 * the associated name (a "measure").
107 * If neither the startMark nor the endMark argument is specified,
108 * measure() will store the duration from $_SERVER['REQUEST_TIME_FLOAT'] to
110 * If the startMark argument is specified, but the endMark argument is not
111 * specified, measure() will store the duration from the most recent
112 * occurrence of the start mark to the current time.
113 * If both the startMark and endMark arguments are specified, measure()
114 * will store the duration from the most recent occurrence of the start
115 * mark to the most recent occurrence of the end mark.
117 * @param string $measureName
118 * @param string $startMark
119 * @param string|null $endMark
120 * @return array|bool The measure that has been created, or false if either
121 * the start mark or the end mark do not exist.
123 public function measure( $measureName, $startMark = 'requestStart', $endMark = null ) {
124 $start = $this->getEntryByName( $startMark );
125 if ( $start === null ) {
126 $this->logger
->error( __METHOD__
. ": The mark '$startMark' does not exist" );
129 $startTime = $start['startTime'];
132 $end = $this->getEntryByName( $endMark );
133 if ( $end === null ) {
134 $this->logger
->error( __METHOD__
. ": The mark '$endMark' does not exist" );
137 $endTime = $end['startTime'];
139 $endTime = microtime( true );
142 $this->entries
[$measureName] = [
143 'name' => $measureName,
144 'entryType' => 'measure',
145 'startTime' => $startTime,
146 'duration' => $endTime - $startTime,
149 return $this->entries
[$measureName];
153 * Sort entries in chronological order with respect to startTime.
155 private function sortEntries() {
156 uasort( $this->entries
, static function ( $a, $b ) {
157 return $a['startTime'] <=> $b['startTime'];
162 * @return array[] All entries in chronological order.
164 public function getEntries() {
165 $this->sortEntries();
166 return $this->entries
;
170 * @param string $entryType
171 * @return array[] Entries (in chronological order) that have the same value
172 * for the entryType attribute as the $entryType parameter.
174 public function getEntriesByType( $entryType ) {
175 $this->sortEntries();
177 foreach ( $this->entries
as $entry ) {
178 if ( $entry['entryType'] === $entryType ) {
186 * @param string $name
187 * @return array|null Entry named $name or null if it does not exist.
189 public function getEntryByName( $name ) {
190 return $this->entries
[$name] ??
null;