friendica/src/Util/DateTimeFormat.php

257 lines
6.4 KiB
PHP
Raw Normal View History

<?php
/**
2022-01-02 02:27:47 -05:00
* @copyright Copyright (C) 2010-2022, the Friendica project
*
* @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/>.
*
*/
namespace Friendica\Util;
2018-10-29 17:20:46 -04:00
use Friendica\Core\Logger;
use DateTime;
use DateTimeZone;
use Exception;
/**
2020-01-19 01:05:23 -05:00
* Temporal class
*/
class DateTimeFormat
{
2021-06-01 18:32:05 -04:00
const ATOM = 'Y-m-d\TH:i:s\Z';
const MYSQL = 'Y-m-d H:i:s';
2021-06-01 18:32:05 -04:00
const HTTP = 'D, d M Y H:i:s \G\M\T';
2021-06-01 23:32:42 -04:00
const JSON = 'Y-m-d\TH:i:s.v\Z';
2021-11-18 16:43:13 -05:00
const API = 'D M d H:i:s +0000 Y';
static $localTimezone = 'UTC';
public static function setLocalTimeZone(string $timezone)
{
self::$localTimezone = $timezone;
}
/**
* 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 16:06:53 -05:00
* @throws Exception
*/
public static function utc(string $time, string $format = self::MYSQL): string
{
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 16:06:53 -05:00
* @throws Exception
*/
public static function local($time, $format = self::MYSQL)
{
return self::convert($time, self::$localTimezone, 'UTC', $format);
}
/**
* convert() shorthand for timezoned now.
*
2019-01-06 16:06:53 -05:00
* @param $timezone
* @param string $format DateTime format string or Temporal constant
* @return string
2019-01-06 16:06:53 -05:00
* @throws Exception
*/
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 16:06:53 -05:00
* @throws Exception
*/
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 16:06:53 -05:00
* @throws Exception
*/
public static function utcNow(string $format = self::MYSQL): string
{
return self::utc('now', $format);
}
/**
2020-01-19 01:05:23 -05:00
* General purpose date parse/convert/format function.
*
* @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 16:06:53 -05:00
* http://www.php.net/manual/en/datetime.format.php
*
* @return string Formatted date according to given format
2019-01-06 16:06:53 -05:00
* @throws Exception
*/
public static function convert($s = 'now', $tz_to = 'UTC', $tz_from = 'UTC', $format = self::MYSQL)
{
// Defaults to UTC if nothing is set, but throws an exception if set to empty string.
// Provide some sane defaults regardless.
if ($tz_from === '') {
$tz_from = 'UTC';
}
if ($tz_to === '') {
$tz_to = 'UTC';
}
if (($s === '') || (!is_string($s))) {
$s = 'now';
}
2022-10-03 12:12:22 -04:00
$s = self::fixDateFormat($s);
/*
* Slight hackish adjustment so that 'zero' datetime actually returns what is intended
* otherwise we end up with -0001-11-30 ...
* add 32 days so that we at least get year 00, and then hack around the fact that
* months and days always start with 1.
*/
if (substr($s, 0, 10) <= '0001-01-01') {
if ($s < '0000-00-00') {
$s = '0000-00-00';
}
$d = new DateTime($s . ' + 32 days', new DateTimeZone('UTC'));
return str_replace('1', '0', $d->format($format));
}
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-08-30 15:45:30 -04:00
Logger::warning('DateTimeFormat::convert: exception: ' . $e->getMessage());
$d = new DateTime('now', $from_obj);
}
try {
$to_obj = new DateTimeZone($tz_to);
} catch (Exception $e) {
$to_obj = new DateTimeZone('UTC');
}
$d->setTimezone($to_obj);
return $d->format($format);
}
2022-10-03 12:12:22 -04:00
/**
* Fix weird date formats
*
* @param string $dateString
* @return string
*/
private static function fixDateFormat(string $dateString): string
{
$patterns = [
['#(\w+), (\d+/\d+/\d+) - (\d+:\d+)#', '$1, $2 $3'],
['#(\d+-\d+-\d+)T(\d+:\d+:\d+)ZZ#', '$1T$2Z'],
['#(\d+-\d+-\d+)T(\d+:\d+:\d+\.\d+)ZZ#', '$1T$2Z'],
['#(\w+), (\d+ \w+ \d+) (\d+:\d+:\d+) (.+)#', '$2 $3 $4'],
['#(\d+:\d+) (\w+), (\w+) (\d+), (\d+)#', '$1 $2 $3 $4 $5'],
2022-10-06 17:09:52 -04:00
['#(\w+ \d+, \d+) - (\d+:\d+)#', '$1, $2'],
2022-10-07 15:56:02 -04:00
['~(\d+-\d+-\d+)T(\d+:\d+:\d+)&#x2B;(\d+:\d+)~', '$1T$2+$3'],
];
foreach ($patterns as $pattern) {
$dateString = preg_replace($pattern[0], $pattern[1], $dateString);
2022-10-03 12:12:22 -04:00
}
2022-10-06 17:09:52 -04:00
2022-10-03 12:12:22 -04:00
return $dateString;
}
/**
* 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;
}
/**
* 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;
}
}