113 lines
2.5 KiB
PHP
113 lines
2.5 KiB
PHP
<?php
|
|
|
|
namespace PicoFeed\Parser;
|
|
|
|
use DateTime;
|
|
use DateTimeZone;
|
|
|
|
/**
|
|
* Date Parser.
|
|
*
|
|
* @author Frederic Guillot
|
|
*/
|
|
class DateParser
|
|
{
|
|
/**
|
|
* Timezone used to parse feed dates.
|
|
*
|
|
* @var string
|
|
*/
|
|
public $timezone = 'UTC';
|
|
|
|
/**
|
|
* Supported formats [ 'format' => length ].
|
|
*
|
|
* @var array
|
|
*/
|
|
public $formats = array(
|
|
DATE_ATOM => null,
|
|
DATE_RSS => null,
|
|
DATE_COOKIE => null,
|
|
DATE_ISO8601 => null,
|
|
DATE_RFC822 => null,
|
|
DATE_RFC850 => null,
|
|
DATE_RFC1036 => null,
|
|
DATE_RFC1123 => null,
|
|
DATE_RFC2822 => null,
|
|
DATE_RFC3339 => null,
|
|
'D, d M Y H:i:s' => 25,
|
|
'D, d M Y h:i:s' => 25,
|
|
'D M d Y H:i:s' => 24,
|
|
'j M Y H:i:s' => 20,
|
|
'Y-m-d H:i:s' => 19,
|
|
'Y-m-d\TH:i:s' => 19,
|
|
'd/m/Y H:i:s' => 19,
|
|
'D, d M Y' => 16,
|
|
'Y-m-d' => 10,
|
|
'd-m-Y' => 10,
|
|
'm-d-Y' => 10,
|
|
'd.m.Y' => 10,
|
|
'm.d.Y' => 10,
|
|
'd/m/Y' => 10,
|
|
'm/d/Y' => 10,
|
|
);
|
|
|
|
/**
|
|
* Try to parse all date format for broken feeds.
|
|
*
|
|
* @param string $value Original date format
|
|
*
|
|
* @return DateTime
|
|
*/
|
|
public function getDateTime($value)
|
|
{
|
|
$value = trim($value);
|
|
|
|
foreach ($this->formats as $format => $length) {
|
|
$truncated_value = $value;
|
|
if ($length !== null) {
|
|
$truncated_value = substr($truncated_value, 0, $length);
|
|
}
|
|
|
|
$date = $this->getValidDate($format, $truncated_value);
|
|
if ($date !== false) {
|
|
return $date;
|
|
}
|
|
}
|
|
|
|
return $this->getCurrentDateTime();
|
|
}
|
|
|
|
/**
|
|
* Get a valid date from a given format.
|
|
*
|
|
* @param string $format Date format
|
|
* @param string $value Original date value
|
|
*
|
|
* @return DateTime|bool
|
|
*/
|
|
public function getValidDate($format, $value)
|
|
{
|
|
$date = DateTime::createFromFormat($format, $value, new DateTimeZone($this->timezone));
|
|
|
|
if ($date !== false) {
|
|
$errors = DateTime::getLastErrors();
|
|
|
|
if ($errors['error_count'] === 0 && $errors['warning_count'] === 0) {
|
|
return $date;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Get the current datetime.
|
|
*
|
|
* @return DateTime
|
|
*/
|
|
public function getCurrentDateTime()
|
|
{
|
|
return new DateTime('now', new DateTimeZone($this->timezone));
|
|
}
|
|
}
|