Merge ".mailmap: Correct two contributor names"
[mediawiki.git] / includes / user / UserTimeCorrection.php
blob533b45552fa12526ba1630272b04ba7f94f0243a
1 <?php
2 /**
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
18 * @file
21 namespace MediaWiki\User;
23 use DateInterval;
24 use DateTime;
25 use DateTimeZone;
26 use Exception;
27 use MediaWiki\Utils\MWTimestamp;
28 use Stringable;
29 use Wikimedia\RequestTimeout\TimeoutException;
31 /**
32 * Utility class to parse the TimeCorrection string value.
34 * These values are used to specify the time offset for a user and are stored in
35 * the database as a user preference and returned by the preferences APIs
37 * The class will correct invalid input and adjusts timezone offsets to applicable dates,
38 * taking into account DST etc.
40 * @since 1.37
41 * @ingroup User
42 * @author Derk-Jan Hartman <hartman.wiki@gmail.com>
44 class UserTimeCorrection implements Stringable {
46 /**
47 * @var string (default) Time correction based on the MediaWiki's system offset from UTC.
48 * The System offset can be configured with wgLocalTimezone and/or wgLocalTZoffset
50 public const SYSTEM = 'System';
52 /** @var string Time correction based on a user defined offset from UTC */
53 public const OFFSET = 'Offset';
55 /** @var string Time correction based on a user defined timezone */
56 public const ZONEINFO = 'ZoneInfo';
58 /** @var DateTime */
59 private $date;
61 /** @var bool */
62 private $valid;
64 /** @var string */
65 private $correctionType;
67 /** @var int Offset in minutes */
68 private $offset;
70 /** @var DateTimeZone|null */
71 private $timeZone;
73 /**
74 * @param string $timeCorrection Original time correction string
75 * @param DateTime|null $relativeToDate The date used to calculate the time zone offset of.
76 * This defaults to the current date and time.
77 * @param int $systemOffset Offset for self::SYSTEM in minutes
79 public function __construct(
80 string $timeCorrection,
81 ?DateTime $relativeToDate = null,
82 int $systemOffset = 0
83 ) {
84 $this->date = $relativeToDate ?? new DateTime( '@' . MWTimestamp::time() );
85 $this->valid = false;
86 $this->parse( $timeCorrection, $systemOffset );
89 /**
90 * Get time offset for a user
92 * @return string Offset that was applied to the user
94 public function getCorrectionType(): string {
95 return $this->correctionType;
98 /**
99 * Get corresponding time offset for this correction
100 * Note: When correcting dates/times, apply only the offset OR the time zone, not both.
101 * @return int Offset in minutes
103 public function getTimeOffset(): int {
104 return $this->offset;
108 * Get corresponding time offset for this correction
109 * Note: When correcting dates/times, apply only the offset OR the time zone, not both.
110 * @return DateInterval Offset in minutes as a DateInterval
112 public function getTimeOffsetInterval(): DateInterval {
113 $offset = abs( $this->offset );
114 $interval = new DateInterval( "PT{$offset}M" );
115 if ( $this->offset < 1 ) {
116 $interval->invert = 1;
118 return $interval;
122 * The time zone if known
123 * Note: When correcting dates/times, apply only the offset OR the time zone, not both.
124 * @return DateTimeZone|null
126 public function getTimeZone(): ?DateTimeZone {
127 return $this->timeZone;
131 * Was the original correction specification valid
132 * @return bool
134 public function isValid(): bool {
135 return $this->valid;
139 * Parse the timecorrection string as stored in the database for a user
140 * or as entered into the Preferences form field
142 * There can be two forms of these strings:
143 * 1. A pipe separated tuple of a maximum of 3 fields
144 * - Field 1 is the type of offset definition
145 * - Field 2 is the offset in minutes from UTC (ignored for System type)
146 * FIXME Since it's ignored, remove the offset from System everywhere.
147 * - Field 3 is a timezone identifier from the tz database (only required for ZoneInfo type)
148 * - The offset for a ZoneInfo type is unreliable because of DST.
149 * After retrieving it from the database, it should be recalculated based on the TZ identifier.
150 * Examples:
151 * - System
152 * - System|60
153 * - Offset|60
154 * - ZoneInfo|60|Europe/Amsterdam
156 * 2. The following form provides an offset in hours and minutes
157 * This currently should only be used by the preferences input field,
158 * but historically they were present in the database.
159 * TODO: write a maintenance script to migrate these old db values
160 * Examples:
161 * - 16:00
162 * - 10
164 * @param string $timeCorrection
165 * @param int $systemOffset
167 private function parse( string $timeCorrection, int $systemOffset ) {
168 $data = explode( '|', $timeCorrection, 3 );
170 // First handle the case of an actual timezone being specified.
171 if ( $data[0] === self::ZONEINFO ) {
172 try {
173 $this->correctionType = self::ZONEINFO;
174 $this->timeZone = new DateTimeZone( $data[2] );
175 $this->offset = (int)floor( $this->timeZone->getOffset( $this->date ) / 60 );
176 $this->valid = true;
177 return;
178 } catch ( TimeoutException $e ) {
179 throw $e;
180 } catch ( Exception $e ) {
181 // Not a valid/known timezone.
182 // Fall back to any specified offset
186 // If $timeCorrection is in fact a pipe-separated value, check the
187 // first value.
188 switch ( $data[0] ) {
189 case self::OFFSET:
190 case self::ZONEINFO:
191 $this->correctionType = self::OFFSET;
192 // First value is Offset, so use the specified offset
193 $this->offset = (int)( $data[1] ?? 0 );
194 // If this is ZoneInfo, then we didn't recognize the TimeZone
195 $this->valid = isset( $data[1] ) && $data[0] === self::OFFSET;
196 break;
197 case self::SYSTEM:
198 $this->correctionType = self::SYSTEM;
199 $this->offset = $systemOffset;
200 $this->valid = true;
201 break;
202 default:
203 // $timeCorrection actually isn't a pipe separated value, but instead
204 // a colon separated value. This is only used by the HTMLTimezoneField userinput
205 // but can also still be present in the Db. (but shouldn't be)
206 $this->correctionType = self::OFFSET;
207 $data = explode( ':', $timeCorrection, 2 );
208 if ( count( $data ) >= 2 ) {
209 // Combination hours and minutes.
210 $this->offset = abs( (int)$data[0] ) * 60 + (int)$data[1];
211 if ( (int)$data[0] < 0 ) {
212 $this->offset *= -1;
214 $this->valid = true;
215 } elseif ( preg_match( '/^[+-]?\d+$/', $data[0] ) ) {
216 // Just hours.
217 $this->offset = (int)$data[0] * 60;
218 $this->valid = true;
219 } else {
220 // We really don't know this. Fallback to System
221 $this->correctionType = self::SYSTEM;
222 $this->offset = $systemOffset;
223 return;
225 break;
228 // Max is +14:00 and min is -12:00, see:
229 // https://en.wikipedia.org/wiki/Timezone
230 if ( $this->offset < -12 * 60 || $this->offset > 14 * 60 ) {
231 $this->valid = false;
233 // 14:00
234 $this->offset = min( $this->offset, 14 * 60 );
235 // -12:00
236 $this->offset = max( $this->offset, -12 * 60 );
240 * Converts a timezone offset in minutes (e.g., "120") to an hh:mm string like "+02:00".
241 * @param int $offset
242 * @return string
244 public static function formatTimezoneOffset( int $offset ): string {
245 $hours = $offset > 0 ? floor( $offset / 60 ) : ceil( $offset / 60 );
246 return sprintf( '%+03d:%02d', $hours, abs( $offset ) % 60 );
250 * Note: The string value of this object might not be equal to the original value
251 * @return string a timecorrection string representing this value
253 public function toString(): string {
254 switch ( $this->correctionType ) {
255 case self::ZONEINFO:
256 if ( $this->timeZone ) {
257 return "ZoneInfo|{$this->offset}|{$this->timeZone->getName()}";
259 // If not, fallback:
260 case self::SYSTEM:
261 case self::OFFSET:
262 default:
263 return "{$this->correctionType}|{$this->offset}";
267 public function __toString() {
268 return $this->toString();