2018-01-27 03:38:34 +01:00
|
|
|
<?php
|
|
|
|
/**
|
2023-01-01 15:36:24 +01:00
|
|
|
* @copyright Copyright (C) 2010-2023, the Friendica project
|
2020-02-09 16:18:46 +01:00
|
|
|
*
|
|
|
|
* @license GNU AGPL version 3 or any later version
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*
|
2018-01-27 03:38:34 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Friendica\Util;
|
|
|
|
|
2018-10-29 22:20:46 +01:00
|
|
|
use Friendica\Core\Logger;
|
2018-01-27 03:38:34 +01:00
|
|
|
use DateTime;
|
|
|
|
use DateTimeZone;
|
|
|
|
use Exception;
|
|
|
|
|
|
|
|
/**
|
2020-01-19 07:05:23 +01:00
|
|
|
* Temporal class
|
2018-01-27 03:38:34 +01:00
|
|
|
*/
|
|
|
|
class DateTimeFormat
|
|
|
|
{
|
2021-06-02 00:32:05 +02:00
|
|
|
const ATOM = 'Y-m-d\TH:i:s\Z';
|
2018-01-27 03:38:34 +01:00
|
|
|
const MYSQL = 'Y-m-d H:i:s';
|
2021-06-02 00:32:05 +02:00
|
|
|
const HTTP = 'D, d M Y H:i:s \G\M\T';
|
2021-06-02 05:32:42 +02:00
|
|
|
const JSON = 'Y-m-d\TH:i:s.v\Z';
|
2021-11-18 22:43:13 +01:00
|
|
|
const API = 'D M d H:i:s +0000 Y';
|
2018-01-27 03:38:34 +01:00
|
|
|
|
2021-10-03 18:38:47 +02:00
|
|
|
static $localTimezone = 'UTC';
|
|
|
|
|
|
|
|
public static function setLocalTimeZone(string $timezone)
|
|
|
|
{
|
|
|
|
self::$localTimezone = $timezone;
|
|
|
|
}
|
|
|
|
|
2018-01-27 03:38:34 +01:00
|
|
|
/**
|
|
|
|
* convert() shorthand for UTC.
|
|
|
|
*
|
|
|
|
* @param string $time A date/time string
|
|
|
|
* @param string $format DateTime format string or Temporal constant
|
|
|
|
* @return string
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws Exception
|
2018-01-27 03:38:34 +01:00
|
|
|
*/
|
2021-12-02 15:07:54 +01:00
|
|
|
public static function utc(string $time, string $format = self::MYSQL): string
|
2018-01-27 03:38:34 +01:00
|
|
|
{
|
|
|
|
return self::convert($time, 'UTC', 'UTC', $format);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* convert() shorthand for local.
|
|
|
|
*
|
|
|
|
* @param string $time A date/time string
|
|
|
|
* @param string $format DateTime format string or Temporal constant
|
|
|
|
* @return string
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws Exception
|
2018-01-27 03:38:34 +01:00
|
|
|
*/
|
|
|
|
public static function local($time, $format = self::MYSQL)
|
|
|
|
{
|
2021-10-03 18:38:47 +02:00
|
|
|
return self::convert($time, self::$localTimezone, 'UTC', $format);
|
2018-01-27 03:38:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* convert() shorthand for timezoned now.
|
|
|
|
*
|
2019-01-06 22:06:53 +01:00
|
|
|
* @param $timezone
|
2018-01-27 03:38:34 +01:00
|
|
|
* @param string $format DateTime format string or Temporal constant
|
|
|
|
* @return string
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws Exception
|
2018-01-27 03:38:34 +01:00
|
|
|
*/
|
|
|
|
public static function timezoneNow($timezone, $format = self::MYSQL)
|
|
|
|
{
|
|
|
|
return self::convert('now', $timezone, 'UTC', $format);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* convert() shorthand for local now.
|
|
|
|
*
|
|
|
|
* @param string $format DateTime format string or Temporal constant
|
|
|
|
* @return string
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws Exception
|
2018-01-27 03:38:34 +01:00
|
|
|
*/
|
|
|
|
public static function localNow($format = self::MYSQL)
|
|
|
|
{
|
|
|
|
return self::local('now', $format);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* convert() shorthand for UTC now.
|
|
|
|
*
|
|
|
|
* @param string $format DateTime format string or Temporal constant
|
|
|
|
* @return string
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws Exception
|
2018-01-27 03:38:34 +01:00
|
|
|
*/
|
2021-12-02 15:07:54 +01:00
|
|
|
public static function utcNow(string $format = self::MYSQL): string
|
2018-01-27 03:38:34 +01:00
|
|
|
{
|
|
|
|
return self::utc('now', $format);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-01-19 07:05:23 +01:00
|
|
|
* General purpose date parse/convert/format function.
|
2018-01-27 03:38:34 +01:00
|
|
|
*
|
|
|
|
* @param string $s Some parseable date/time string
|
|
|
|
* @param string $tz_to Destination timezone
|
|
|
|
* @param string $tz_from Source timezone
|
|
|
|
* @param string $format Output format recognised from php's DateTime class
|
2019-01-06 22:06:53 +01:00
|
|
|
* http://www.php.net/manual/en/datetime.format.php
|
2018-01-27 03:38:34 +01:00
|
|
|
*
|
|
|
|
* @return string Formatted date according to given format
|
2019-01-06 22:06:53 +01:00
|
|
|
* @throws Exception
|
2018-01-27 03:38:34 +01:00
|
|
|
*/
|
2022-10-09 06:23:09 +02:00
|
|
|
public static function convert(string $s = 'now', string $tz_to = 'UTC', string $tz_from = 'UTC', string $format = self::MYSQL): string
|
2018-01-27 03:38:34 +01:00
|
|
|
{
|
|
|
|
// Defaults to UTC if nothing is set, but throws an exception if set to empty string.
|
|
|
|
// Provide some sane defaults regardless.
|
2018-02-12 16:08:28 +01:00
|
|
|
if ($tz_from === '') {
|
|
|
|
$tz_from = 'UTC';
|
2018-01-27 03:38:34 +01:00
|
|
|
}
|
|
|
|
|
2018-02-12 16:08:28 +01:00
|
|
|
if ($tz_to === '') {
|
|
|
|
$tz_to = 'UTC';
|
2018-01-27 03:38:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (($s === '') || (!is_string($s))) {
|
|
|
|
$s = 'now';
|
|
|
|
}
|
|
|
|
|
2022-10-09 06:23:09 +02:00
|
|
|
// Lowest possible datetime value
|
2018-01-27 03:38:34 +01:00
|
|
|
if (substr($s, 0, 10) <= '0001-01-01') {
|
2022-10-09 06:23:09 +02:00
|
|
|
$d = new DateTime('now', new DateTimeZone('UTC'));
|
|
|
|
$d->setDate(1, 1, 1)->setTime(0, 0);
|
|
|
|
return $d->format($format);
|
2018-01-27 03:38:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
$from_obj = new DateTimeZone($tz_from);
|
|
|
|
} catch (Exception $e) {
|
|
|
|
$from_obj = new DateTimeZone('UTC');
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
$d = new DateTime($s, $from_obj);
|
|
|
|
} catch (Exception $e) {
|
2022-10-09 06:23:09 +02:00
|
|
|
try {
|
|
|
|
$d = new DateTime(self::fix($s), $from_obj);
|
|
|
|
} catch (\Throwable $e) {
|
|
|
|
Logger::warning('DateTimeFormat::convert: exception: ' . $e->getMessage());
|
|
|
|
$d = new DateTime('now', $from_obj);
|
|
|
|
}
|
2018-01-27 03:38:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
$to_obj = new DateTimeZone($tz_to);
|
|
|
|
} catch (Exception $e) {
|
|
|
|
$to_obj = new DateTimeZone('UTC');
|
|
|
|
}
|
|
|
|
|
2021-10-03 18:38:47 +02:00
|
|
|
$d->setTimezone($to_obj);
|
2018-01-27 03:38:34 +01:00
|
|
|
|
|
|
|
return $d->format($format);
|
|
|
|
}
|
2019-10-23 02:39:28 +02:00
|
|
|
|
2022-10-03 18:12:22 +02:00
|
|
|
/**
|
2022-10-09 06:23:09 +02:00
|
|
|
* Fix weird date formats.
|
|
|
|
*
|
|
|
|
* Note: This method isn't meant to sanitize valid date/time strings, for example it will mangle relative date
|
|
|
|
* strings like "now - 3 days".
|
2022-10-03 18:12:22 +02:00
|
|
|
*
|
2022-10-07 23:33:52 +02:00
|
|
|
* @see \Friendica\Test\src\Util\DateTimeFormatTest::dataFix() for a list of examples handled by this method.
|
2022-10-03 18:12:22 +02:00
|
|
|
* @param string $dateString
|
|
|
|
* @return string
|
|
|
|
*/
|
2022-10-07 23:32:34 +02:00
|
|
|
public static function fix(string $dateString): string
|
2022-10-03 18:12:22 +02:00
|
|
|
{
|
2023-01-09 14:51:20 +01:00
|
|
|
$search = ['Mär', 'März', 'Mai', 'Juni', 'Juli', 'Okt', 'Dez', 'ET' , 'ZZ', ' - ', '+', '&#43;', ' (Coordinated Universal Time)'];
|
|
|
|
$replace = ['Mar', 'Mar' , 'May', 'Jun' , 'Jul' , 'Oct', 'Dec', 'EST', 'Z' , ', ' , '+' , '+' , ''];
|
2022-10-09 15:24:06 +02:00
|
|
|
|
|
|
|
$dateString = str_replace($search, $replace, $dateString);
|
|
|
|
|
|
|
|
$pregPatterns = [
|
2022-10-03 19:04:49 +02:00
|
|
|
['#(\w+), (\d+ \w+ \d+) (\d+:\d+:\d+) (.+)#', '$2 $3 $4'],
|
|
|
|
['#(\d+:\d+) (\w+), (\w+) (\d+), (\d+)#', '$1 $2 $3 $4 $5'],
|
|
|
|
];
|
|
|
|
|
2022-10-09 15:24:06 +02:00
|
|
|
foreach ($pregPatterns as $pattern) {
|
2022-10-03 19:04:49 +02:00
|
|
|
$dateString = preg_replace($pattern[0], $pattern[1], $dateString);
|
2022-10-03 18:12:22 +02:00
|
|
|
}
|
2022-10-06 23:09:52 +02:00
|
|
|
|
2022-10-03 18:12:22 +02:00
|
|
|
return $dateString;
|
|
|
|
}
|
|
|
|
|
2019-10-23 02:39:28 +02:00
|
|
|
/**
|
|
|
|
* Checks, if the given string is a date with the pattern YYYY-MM
|
|
|
|
*
|
|
|
|
* @param string $dateString The given date
|
|
|
|
*
|
|
|
|
* @return boolean True, if the date is a valid pattern
|
|
|
|
*/
|
|
|
|
public function isYearMonth(string $dateString)
|
|
|
|
{
|
|
|
|
// Check format (2019-01, 2019-1, 2019-10)
|
|
|
|
if (!preg_match('/^([12]\d{3}-(1[0-2]|0[1-9]|\d))$/', $dateString)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$date = DateTime::createFromFormat('Y-m', $dateString);
|
|
|
|
|
|
|
|
if (!$date) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
$now = new DateTime();
|
|
|
|
} catch (\Throwable $t) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($date > $now) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2020-01-15 05:06:30 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks, if the given string is a date with the pattern YYYY-MM-DD
|
|
|
|
*
|
|
|
|
* @param string $dateString The given date
|
|
|
|
*
|
|
|
|
* @return boolean True, if the date is a valid pattern
|
|
|
|
*/
|
|
|
|
public function isYearMonthDay(string $dateString)
|
|
|
|
{
|
|
|
|
$date = DateTime::createFromFormat('Y-m-d', $dateString);
|
|
|
|
if (!$date) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (DateTime::getLastErrors()['error_count'] || DateTime::getLastErrors()['warning_count']) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2018-01-27 03:38:34 +01:00
|
|
|
}
|