179 lines
4.3 KiB
PHP
Raw Normal View History

2013-02-17 21:48:21 -05:00
<?php
namespace PicoFeed;
class Reader
{
private $url = '';
private $content = '';
public function __construct($content = '')
{
$this->content = $content;
return $this;
}
public function download($url, $timeout = 5, $user_agent = 'PicoFeed (https://github.com/fguillot/picoFeed)')
2013-02-17 21:48:21 -05:00
{
if (strpos($url, 'http') !== 0) {
$url = 'http://'.$url;
}
$this->url = $url;
$this->content = $this->fetchRemoteFile($url, $timeout, $user_agent);
2013-02-17 21:48:21 -05:00
return $this;
}
public function fetchRemoteFile($url, $timeout, $user_agent)
{
if (! \function_exists('curl_init')) {
return @file_get_contents($this->url);
}
$ch = \curl_init();
\curl_setopt($ch, CURLOPT_URL, $url);
\curl_setopt($ch, CURLOPT_HEADER, false);
\curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
\curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, $timeout);
\curl_setopt($ch, CURLOPT_USERAGENT, $user_agent);
$content = \curl_exec($ch);
\curl_close($ch);
return $content;
}
2013-02-17 21:48:21 -05:00
public function getContent()
{
return $this->content;
}
public function getUrl()
{
return $this->url;
}
2013-02-24 14:03:14 -05:00
public function getFirstTag($data)
2013-02-17 21:48:21 -05:00
{
2013-03-21 19:58:52 -04:00
// Strip HTML comments
$data = preg_replace('/<!--(.*)-->/Uis', '', $data);
// Find <?xml version....
2013-02-24 14:03:14 -05:00
if (strpos($data, '<?xml') !== false) {
2013-02-17 21:48:21 -05:00
2013-02-24 14:03:14 -05:00
$data = substr($data, strrpos($data, '?>') + 2);
2013-02-17 21:48:21 -05:00
2013-03-21 19:58:52 -04:00
// Find the first tag
2013-02-24 14:03:14 -05:00
$open_tag = strpos($data, '<');
$close_tag = strpos($data, '>');
2013-02-17 21:48:21 -05:00
2013-02-24 14:03:14 -05:00
return substr($data, $open_tag, $close_tag);
2013-02-17 21:48:21 -05:00
}
2013-02-24 14:03:14 -05:00
return $data;
}
public function getParser($discover = false)
{
$first_tag = $this->getFirstTag($this->content);
if (strpos($first_tag, '<feed ') !== false) {
2013-02-17 21:48:21 -05:00
2013-04-04 23:34:07 -04:00
require_once __DIR__.'/Parsers/Atom.php';
2013-02-17 21:48:21 -05:00
return new Atom($this->content);
}
2013-04-01 21:31:54 -04:00
else if (strpos($first_tag, '<rss ') !== false &&
(strpos($first_tag, 'version="2.0"') !== false || strpos($first_tag, 'version=\'2.0\'') !== false)) {
2013-02-17 21:48:21 -05:00
2013-04-04 23:34:07 -04:00
require_once __DIR__.'/Parsers/Rss20.php';
2013-02-17 21:48:21 -05:00
return new Rss20($this->content);
2013-02-24 14:03:14 -05:00
}
2013-04-01 21:31:54 -04:00
else if (strpos($first_tag, '<rss ') !== false &&
(strpos($first_tag, 'version="0.92"') !== false || strpos($first_tag, 'version=\'0.92\'') !== false)) {
2013-03-25 21:29:55 -04:00
2013-04-04 23:34:07 -04:00
require_once __DIR__.'/Parsers/Rss92.php';
2013-03-25 21:29:55 -04:00
return new Rss92($this->content);
}
2013-04-01 21:31:54 -04:00
else if (strpos($first_tag, '<rss ') !== false &&
(strpos($first_tag, 'version="0.91"') !== false || strpos($first_tag, 'version=\'0.91\'') !== false)) {
2013-03-25 21:29:55 -04:00
2013-04-04 23:34:07 -04:00
require_once __DIR__.'/Parsers/Rss91.php';
2013-03-25 21:29:55 -04:00
return new Rss91($this->content);
}
2013-03-20 00:19:12 -04:00
else if (strpos($first_tag, '<rdf:') !== false && strpos($first_tag, 'xmlns="http://purl.org/rss/1.0/"') !== false) {
2013-04-04 23:34:07 -04:00
require_once __DIR__.'/Parsers/Rss10.php';
2013-03-20 00:19:12 -04:00
return new Rss10($this->content);
}
2013-02-24 14:03:14 -05:00
else if ($discover === true) {
return false;
}
else if ($this->discover()) {
2013-02-17 21:48:21 -05:00
2013-02-24 14:03:14 -05:00
return $this->getParser(true);
}
2013-02-17 21:48:21 -05:00
return false;
}
public function discover()
{
2013-02-24 14:03:14 -05:00
if (! $this->content) {
return false;
}
2013-02-17 21:48:21 -05:00
\libxml_use_internal_errors(true);
$dom = new \DOMDocument;
$dom->loadHTML($this->content);
$xpath = new \DOMXPath($dom);
$queries = array(
"//link[@type='application/atom+xml']",
"//link[@type='application/rss+xml']"
);
foreach ($queries as $query) {
$nodes = $xpath->query($query);
if ($nodes->length !== 0) {
$link = $nodes->item(0)->getAttribute('href');
// Relative links
if (strpos($link, 'http') !== 0) {
if ($link{0} === '/') $link = substr($link, 1);
if ($this->url{strlen($this->url) - 1} !== '/') $this->url .= '/';
$link = $this->url.$link;
}
$this->download($link);
return true;
}
}
return false;
}
}