2013-04-04 23:34:07 -04:00
|
|
|
<?php
|
|
|
|
|
2014-12-23 21:28:26 -05:00
|
|
|
namespace PicoFeed\Parser;
|
2013-04-04 23:34:07 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
use SimpleXMLElement;
|
2014-12-23 21:28:26 -05:00
|
|
|
use PicoFeed\Filter\Filter;
|
|
|
|
use PicoFeed\Client\Url;
|
2014-05-20 14:20:27 -04:00
|
|
|
|
2014-03-16 21:35:57 -04:00
|
|
|
/**
|
|
|
|
* RSS 2.0 Parser
|
|
|
|
*
|
|
|
|
* @author Frederic Guillot
|
2014-12-23 21:28:26 -05:00
|
|
|
* @package Parser
|
2014-03-16 21:35:57 -04:00
|
|
|
*/
|
2014-05-20 14:20:27 -04:00
|
|
|
class Rss20 extends Parser
|
2013-04-04 23:34:07 -04:00
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
/**
|
|
|
|
* Supported namespaces
|
|
|
|
*/
|
|
|
|
protected $namespaces = array(
|
|
|
|
'dc' => 'http://purl.org/dc/elements/1.1/',
|
|
|
|
'content' => 'http://purl.org/rss/1.0/modules/content/',
|
|
|
|
'feedburner' => 'http://rssnamespace.org/feedburner/ext/1.0',
|
|
|
|
'atom' => 'http://www.w3.org/2005/Atom'
|
|
|
|
);
|
|
|
|
|
2014-03-16 21:35:57 -04:00
|
|
|
/**
|
2014-05-20 14:20:27 -04:00
|
|
|
* Get the path to the items XML tree
|
2014-03-16 21:35:57 -04:00
|
|
|
*
|
|
|
|
* @access public
|
2014-05-20 14:20:27 -04:00
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @return SimpleXMLElement
|
2014-03-16 21:35:57 -04:00
|
|
|
*/
|
2014-05-20 14:20:27 -04:00
|
|
|
public function getItemsTree(SimpleXMLElement $xml)
|
2013-04-04 23:34:07 -04:00
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
return XmlParser::getXPathResult($xml, 'channel/item');
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2013-04-04 23:34:07 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the feed url
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findFeedUrl(SimpleXMLElement $xml, Feed $feed)
|
|
|
|
{
|
2014-12-23 21:28:26 -05:00
|
|
|
$feed->feed_url = '';
|
|
|
|
}
|
2013-04-12 15:57:54 -04:00
|
|
|
|
2014-12-23 21:28:26 -05:00
|
|
|
/**
|
|
|
|
* Find the site url
|
|
|
|
*
|
|
|
|
* @access public
|
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
|
|
|
*/
|
|
|
|
public function findSiteUrl(SimpleXMLElement $xml, Feed $feed)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$site_url = XmlParser::getXPathResult($xml, 'channel/link');
|
|
|
|
$feed->site_url = (string) current($site_url);
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2013-04-12 15:57:54 -04:00
|
|
|
|
2014-10-19 14:42:31 -04:00
|
|
|
/**
|
|
|
|
* Find the feed description
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-10-19 14:42:31 -04:00
|
|
|
*/
|
|
|
|
public function findFeedDescription(SimpleXMLElement $xml, Feed $feed)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$description = XmlParser::getXPathResult($xml, 'channel/description');
|
|
|
|
$feed->description = (string) current($description);
|
2014-10-19 14:42:31 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find the feed logo url
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-10-19 14:42:31 -04:00
|
|
|
*/
|
|
|
|
public function findFeedLogo(SimpleXMLElement $xml, Feed $feed)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$logo = XmlParser::getXPathResult($xml, 'channel/image/url');
|
|
|
|
$feed->logo = (string) current($logo);
|
2014-10-19 14:42:31 -04:00
|
|
|
}
|
|
|
|
|
2015-01-18 15:20:36 +01:00
|
|
|
/**
|
|
|
|
* Find the feed icon
|
|
|
|
*
|
|
|
|
* @access public
|
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
|
|
|
*/
|
|
|
|
public function findFeedIcon(SimpleXMLElement $xml, Feed $feed)
|
|
|
|
{
|
|
|
|
$feed->icon = '';
|
|
|
|
}
|
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the feed title
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findFeedTitle(SimpleXMLElement $xml, Feed $feed)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$title = XmlParser::getXPathResult($xml, 'channel/title');
|
|
|
|
$feed->title = Filter::stripWhiteSpace((string) current($title)) ?: $feed->getSiteUrl();
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2013-04-04 23:34:07 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the feed language
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findFeedLanguage(SimpleXMLElement $xml, Feed $feed)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$language = XmlParser::getXPathResult($xml, 'channel/language');
|
|
|
|
$feed->language = (string) current($language);
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2014-01-11 19:46:30 -05:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the feed id
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findFeedId(SimpleXMLElement $xml, Feed $feed)
|
|
|
|
{
|
2014-12-23 21:28:26 -05:00
|
|
|
$feed->id = $feed->getFeedUrl() ?: $feed->getSiteUrl();
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2013-06-26 19:30:46 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the feed date
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $xml Feed xml
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findFeedDate(SimpleXMLElement $xml, Feed $feed)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$publish_date = XmlParser::getXPathResult($xml, 'channel/pubDate');
|
|
|
|
$update_date = XmlParser::getXPathResult($xml, 'channel/lastBuildDate');
|
|
|
|
|
|
|
|
$published = ! empty($publish_date) ? $this->date->getDateTime((string) current($publish_date)) : null;
|
|
|
|
$updated = ! empty($update_date) ? $this->date->getDateTime((string) current($update_date)) : null;
|
|
|
|
|
|
|
|
if ($published === null && $updated === null) {
|
|
|
|
$feed->date = $this->date->getCurrentDateTime(); // We use the current date if there is no date for the feed
|
|
|
|
}
|
|
|
|
else if ($published !== null && $updated !== null) {
|
|
|
|
$feed->date = max($published, $updated); // We use the most recent date between published and updated
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$feed->date = $updated ?: $published;
|
|
|
|
}
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2013-04-04 23:34:07 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the item date
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $entry Feed item
|
2015-03-02 22:31:20 -05:00
|
|
|
* @param Item $item Item object
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
2015-03-02 22:31:20 -05:00
|
|
|
public function findItemDate(SimpleXMLElement $entry, Item $item, Feed $feed)
|
2014-05-20 14:20:27 -04:00
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$date = XmlParser::getXPathResult($entry, 'pubDate');
|
2013-04-04 23:34:07 -04:00
|
|
|
|
2015-07-19 11:19:26 -04:00
|
|
|
$item->date = empty($date) ? $feed->getDate() : $this->date->getDateTime((string) current($date));
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2014-02-17 22:04:49 -05:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the item title
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $entry Feed item
|
|
|
|
* @param \PicoFeed\Parser\Item $item Item object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findItemTitle(SimpleXMLElement $entry, Item $item)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$title = XmlParser::getXPathResult($entry, 'title');
|
|
|
|
$item->title = Filter::stripWhiteSpace((string) current($title)) ?: $item->url;
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2013-06-29 13:41:36 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the item author
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $xml Feed
|
|
|
|
* @param SimpleXMLElement $entry Feed item
|
|
|
|
* @param \PicoFeed\Parser\Item $item Item object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findItemAuthor(SimpleXMLElement $xml, SimpleXMLElement $entry, Item $item)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$author = XmlParser::getXPathResult($entry, 'dc:creator', $this->namespaces)
|
|
|
|
?: XmlParser::getXPathResult($entry, 'author')
|
|
|
|
?: XmlParser::getXPathResult($xml, 'channel/dc:creator', $this->namespaces)
|
|
|
|
?: XmlParser::getXPathResult($xml, 'channel/managingEditor');
|
2014-05-20 14:20:27 -04:00
|
|
|
|
2015-07-19 11:19:26 -04:00
|
|
|
$item->author = (string) current($author);
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2013-06-29 13:41:36 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the item content
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $entry Feed item
|
|
|
|
* @param \PicoFeed\Parser\Item $item Item object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findItemContent(SimpleXMLElement $entry, Item $item)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$content = XmlParser::getXPathResult($entry, 'content:encoded', $this->namespaces);
|
2013-04-04 23:34:07 -04:00
|
|
|
|
2015-07-19 11:19:26 -04:00
|
|
|
if (trim((string) current($content)) === '') {
|
|
|
|
$content = XmlParser::getXPathResult($entry, 'description');
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2013-04-04 23:34:07 -04:00
|
|
|
|
2015-07-19 11:19:26 -04:00
|
|
|
$item->content = (string) current($content);
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2013-04-04 23:34:07 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the item URL
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $entry Feed item
|
|
|
|
* @param \PicoFeed\Parser\Item $item Item object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findItemUrl(SimpleXMLElement $entry, Item $item)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$link = XmlParser::getXPathResult($entry, 'feedburner:origLink', $this->namespaces)
|
|
|
|
?: XmlParser::getXPathResult($entry, 'link')
|
|
|
|
?: XmlParser::getXPathResult($entry, 'atom:link/@href', $this->namespaces);
|
|
|
|
|
|
|
|
if (! empty($link)) {
|
|
|
|
$item->url = trim((string) current($link));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$link = XmlParser::getXPathResult($entry, 'guid');
|
|
|
|
$link = trim((string) current($link));
|
|
|
|
|
|
|
|
if (filter_var($link, FILTER_VALIDATE_URL) !== false) {
|
2014-05-24 12:21:07 -04:00
|
|
|
$item->url = $link;
|
2013-04-07 10:58:46 -04:00
|
|
|
}
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
|
|
|
}
|
2013-04-04 23:34:07 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Genereate the item id
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $entry Feed item
|
|
|
|
* @param \PicoFeed\Parser\Item $item Item object
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findItemId(SimpleXMLElement $entry, Item $item, Feed $feed)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$id = (string) current(XmlParser::getXPathResult($entry, 'guid'));
|
2014-02-17 22:04:49 -05:00
|
|
|
|
2014-12-23 21:28:26 -05:00
|
|
|
if ($id) {
|
|
|
|
$item->id = $this->generateId($id);
|
2014-05-28 19:16:36 -04:00
|
|
|
}
|
|
|
|
else {
|
2014-12-23 21:28:26 -05:00
|
|
|
$item->id = $this->generateId(
|
|
|
|
$item->getTitle(), $item->getUrl(), $item->getContent()
|
|
|
|
);
|
2014-05-28 19:16:36 -04:00
|
|
|
}
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
2014-02-17 22:04:49 -05:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the item enclosure
|
|
|
|
*
|
|
|
|
* @access public
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param SimpleXMLElement $entry Feed item
|
|
|
|
* @param \PicoFeed\Parser\Item $item Item object
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findItemEnclosure(SimpleXMLElement $entry, Item $item, Feed $feed)
|
|
|
|
{
|
|
|
|
if (isset($entry->enclosure)) {
|
2015-07-19 11:19:26 -04:00
|
|
|
$enclosure_url = XmlParser::getXPathResult($entry, 'feedburner:origEnclosureLink', $this->namespaces)
|
|
|
|
?: XmlParser::getXPathResult($entry, 'enclosure/@url');
|
2014-02-17 22:04:49 -05:00
|
|
|
|
2015-07-19 11:19:26 -04:00
|
|
|
$enclosure_type = XmlParser::getXPathResult($entry, 'enclosure/@type');
|
2014-05-20 14:20:27 -04:00
|
|
|
|
2014-02-17 10:41:22 -08:00
|
|
|
|
2015-07-19 11:19:26 -04:00
|
|
|
$item->enclosure_url = Url::resolve((string) current($enclosure_url), $feed->getSiteUrl());
|
|
|
|
$item->enclosure_type = (string) current($enclosure_type);
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|
|
|
|
}
|
2013-08-29 19:34:11 -04:00
|
|
|
|
2014-05-20 14:20:27 -04:00
|
|
|
/**
|
|
|
|
* Find the item language
|
|
|
|
*
|
|
|
|
* @access public
|
|
|
|
* @param SimpleXMLElement $entry Feed item
|
2014-12-23 21:28:26 -05:00
|
|
|
* @param \PicoFeed\Parser\Item $item Item object
|
|
|
|
* @param \PicoFeed\Parser\Feed $feed Feed object
|
2014-05-20 14:20:27 -04:00
|
|
|
*/
|
|
|
|
public function findItemLanguage(SimpleXMLElement $entry, Item $item, Feed $feed)
|
|
|
|
{
|
2015-07-19 11:19:26 -04:00
|
|
|
$language = XmlParser::getXPathResult($entry, 'dc:language', $this->namespaces);
|
|
|
|
|
|
|
|
$item->language = (string) current($language) ?: $feed->language;
|
2013-04-04 23:34:07 -04:00
|
|
|
}
|
2014-05-20 14:20:27 -04:00
|
|
|
}
|