forked from GNUsocial/gnu-social
		
	* renamed FeedSub plugin to OStatus * now setting avatar on subscriptions * general fixes for subscription * integrated PuSH hub to handle only user timelines on canonical ID url; sends updates directly * set $config['feedsub']['nohub'] = true to test w/ foreign feeds that don't have hubs (won't actually receive updates though) * a few bits of code documentation * HMAC support for verified distributions (safest if sub setup is on HTTPS) And a couple core changes: * minimizing HTML output for exceptions in API requests to aid in debugging * fix for rel=self link in apitimelineuser when id given This does not not yet include any of the individual subscription management (Salmon notifications for sub/unsub, etc) nor a nice UI for user subscriptions. Needs some further cleanup to treat posts as status updates instead of link references.
		
			
				
	
	
		
			351 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			351 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
	
	
| <?php
 | |
| /* vim: set expandtab tabstop=4 shiftwidth=4 softtabstop=4: */
 | |
| 
 | |
| /**
 | |
|  * Key gateway class for XML_Feed_Parser package
 | |
|  *
 | |
|  * PHP versions 5
 | |
|  *
 | |
|  * LICENSE: This source file is subject to version 3.0 of the PHP license
 | |
|  * that is available through the world-wide-web at the following URI:
 | |
|  * http://www.php.net/license/3_0.txt.  If you did not receive a copy of
 | |
|  * the PHP License and are unable to obtain it through the web, please
 | |
|  * send a note to license@php.net so we can mail you a copy immediately.
 | |
|  *
 | |
|  * @category   XML
 | |
|  * @package    XML_Feed_Parser
 | |
|  * @author     James Stewart <james@jystewart.net>
 | |
|  * @copyright  2005 James Stewart <james@jystewart.net>
 | |
|  * @license    http://www.gnu.org/copyleft/lesser.html  GNU LGPL
 | |
|  * @version    CVS: $Id: Parser.php,v 1.24 2006/08/15 13:04:00 jystewart Exp $
 | |
|  * @link       http://pear.php.net/package/XML_Feed_Parser/
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * XML_Feed_Parser_Type is an abstract class required by all of our
 | |
|  * feed types. It makes sense to load it here to keep the other files
 | |
|  * clean.
 | |
|  */
 | |
| require_once 'XML/Feed/Parser/Type.php';
 | |
| 
 | |
| /**
 | |
|  * We will throw exceptions when errors occur.
 | |
|  */
 | |
| require_once 'XML/Feed/Parser/Exception.php';
 | |
| 
 | |
| /**
 | |
|  * This is the core of the XML_Feed_Parser package. It identifies feed types 
 | |
|  * and abstracts access to them. It is an iterator, allowing for easy access 
 | |
|  * to the entire feed.
 | |
|  *
 | |
|  * @author  James Stewart <james@jystewart.net>
 | |
|  * @version Release: 1.0.3
 | |
|  * @package XML_Feed_Parser
 | |
|  */
 | |
| class XML_Feed_Parser implements Iterator
 | |
| {
 | |
|     /**
 | |
|      * This is where we hold the feed object 
 | |
|      * @var Object
 | |
|      */
 | |
|     private $feed;
 | |
| 
 | |
|     /**
 | |
|      * To allow for extensions, we make a public reference to the feed model 
 | |
|      * @var DOMDocument
 | |
|      */
 | |
|     public $model;
 | |
|     
 | |
|     /**
 | |
|      * A map between entry ID and offset
 | |
|      * @var array
 | |
|      */
 | |
|     protected $idMappings = array();
 | |
| 
 | |
|     /**
 | |
|      * A storage space for Namespace URIs.
 | |
|      * @var array
 | |
|      */
 | |
|     private $feedNamespaces = array(
 | |
|         'rss2' => array(
 | |
|             'http://backend.userland.com/rss',
 | |
|             'http://backend.userland.com/rss2',
 | |
|             'http://blogs.law.harvard.edu/tech/rss'));
 | |
|     /**
 | |
|      * Detects feed types and instantiate appropriate objects.
 | |
|      *
 | |
|      * Our constructor takes care of detecting feed types and instantiating
 | |
|      * appropriate classes. For now we're going to treat Atom 0.3 as Atom 1.0
 | |
|      * but raise a warning. I do not intend to introduce full support for 
 | |
|      * Atom 0.3 as it has been deprecated, but others are welcome to.
 | |
|      *
 | |
|      * @param    string    $feed    XML serialization of the feed
 | |
|      * @param    bool    $strict    Whether or not to validate the feed
 | |
|      * @param    bool    $suppressWarnings Trigger errors for deprecated feed types?
 | |
|      * @param    bool    $tidy    Whether or not to try and use the tidy library on input
 | |
|      */
 | |
|     function __construct($feed, $strict = false, $suppressWarnings = false, $tidy = false)
 | |
|     {
 | |
|         $this->model = new DOMDocument;
 | |
|         if (! $this->model->loadXML($feed)) {
 | |
|             if (extension_loaded('tidy') && $tidy) {
 | |
|                 $tidy = new tidy;
 | |
|                 $tidy->parseString($feed, 
 | |
|                     array('input-xml' => true, 'output-xml' => true));
 | |
|                 $tidy->cleanRepair();
 | |
|                 if (! $this->model->loadXML((string) $tidy)) {
 | |
|                     throw new XML_Feed_Parser_Exception('Invalid input: this is not ' .
 | |
|                         'valid XML');
 | |
|                 }
 | |
|             } else {
 | |
|                 throw new XML_Feed_Parser_Exception('Invalid input: this is not valid XML');
 | |
|             }
 | |
| 
 | |
|         }
 | |
| 
 | |
|         /* detect feed type */
 | |
|         $doc_element = $this->model->documentElement;
 | |
|         $error = false;
 | |
| 
 | |
|         switch (true) {
 | |
|             case ($doc_element->namespaceURI == 'http://www.w3.org/2005/Atom'):
 | |
|                 require_once 'XML/Feed/Parser/Atom.php';
 | |
|                 require_once 'XML/Feed/Parser/AtomElement.php';
 | |
|                 $class = 'XML_Feed_Parser_Atom';
 | |
|                 break;
 | |
|             case ($doc_element->namespaceURI == 'http://purl.org/atom/ns#'):
 | |
|                 require_once 'XML/Feed/Parser/Atom.php';
 | |
|                 require_once 'XML/Feed/Parser/AtomElement.php';
 | |
|                 $class = 'XML_Feed_Parser_Atom';
 | |
|                 $error = 'Atom 0.3 deprecated, using 1.0 parser which won\'t provide ' .
 | |
|                     'all options';
 | |
|                 break;
 | |
|             case ($doc_element->namespaceURI == 'http://purl.org/rss/1.0/' || 
 | |
|                 ($doc_element->hasChildNodes() && $doc_element->childNodes->length > 1 
 | |
|                 && $doc_element->childNodes->item(1)->namespaceURI == 
 | |
|                 'http://purl.org/rss/1.0/')):
 | |
|                 require_once 'XML/Feed/Parser/RSS1.php';
 | |
|                 require_once 'XML/Feed/Parser/RSS1Element.php';
 | |
|                 $class = 'XML_Feed_Parser_RSS1';
 | |
|                 break;
 | |
|             case ($doc_element->namespaceURI == 'http://purl.org/rss/1.1/' || 
 | |
|                 ($doc_element->hasChildNodes() && $doc_element->childNodes->length > 1 
 | |
|                 && $doc_element->childNodes->item(1)->namespaceURI == 
 | |
|                 'http://purl.org/rss/1.1/')):
 | |
|                 require_once 'XML/Feed/Parser/RSS11.php';
 | |
|                 require_once 'XML/Feed/Parser/RSS11Element.php';
 | |
|                 $class = 'XML_Feed_Parser_RSS11';
 | |
|                 break;
 | |
|             case (($doc_element->hasChildNodes() && $doc_element->childNodes->length > 1
 | |
|                 && $doc_element->childNodes->item(1)->namespaceURI == 
 | |
|                 'http://my.netscape.com/rdf/simple/0.9/') || 
 | |
|                 $doc_element->namespaceURI == 'http://my.netscape.com/rdf/simple/0.9/'):
 | |
|                 require_once 'XML/Feed/Parser/RSS09.php';
 | |
|                 require_once 'XML/Feed/Parser/RSS09Element.php';
 | |
|                 $class = 'XML_Feed_Parser_RSS09';
 | |
|                 break;
 | |
|             case ($doc_element->tagName == 'rss' and
 | |
|                 $doc_element->hasAttribute('version') && 
 | |
|                 $doc_element->getAttribute('version') == 0.91):
 | |
|                 $error = 'RSS 0.91 has been superceded by RSS2.0. Using RSS2.0 parser.';
 | |
|                 require_once 'XML/Feed/Parser/RSS2.php';
 | |
|                 require_once 'XML/Feed/Parser/RSS2Element.php';
 | |
|                 $class = 'XML_Feed_Parser_RSS2';
 | |
|                 break;
 | |
|             case ($doc_element->tagName == 'rss' and
 | |
|                 $doc_element->hasAttribute('version') && 
 | |
|                 $doc_element->getAttribute('version') == 0.92):
 | |
|                 $error = 'RSS 0.92 has been superceded by RSS2.0. Using RSS2.0 parser.';
 | |
|                 require_once 'XML/Feed/Parser/RSS2.php';
 | |
|                 require_once 'XML/Feed/Parser/RSS2Element.php';
 | |
|                 $class = 'XML_Feed_Parser_RSS2';
 | |
|                 break;
 | |
|             case (in_array($doc_element->namespaceURI, $this->feedNamespaces['rss2'])
 | |
|                 || $doc_element->tagName == 'rss'):
 | |
|                 if (! $doc_element->hasAttribute('version') || 
 | |
|                     $doc_element->getAttribute('version') != 2) {
 | |
|                     $error = 'RSS version not specified. Parsing as RSS2.0';
 | |
|                 }
 | |
|                 require_once 'XML/Feed/Parser/RSS2.php';
 | |
|                 require_once 'XML/Feed/Parser/RSS2Element.php';
 | |
|                 $class = 'XML_Feed_Parser_RSS2';
 | |
|                 break;
 | |
|             default:
 | |
|                 throw new XML_Feed_Parser_Exception('Feed type unknown');
 | |
|                 break;
 | |
|         }
 | |
| 
 | |
|         if (! $suppressWarnings && ! empty($error)) {
 | |
|             trigger_error($error, E_USER_WARNING);
 | |
|         }
 | |
| 
 | |
|         /* Instantiate feed object */
 | |
|         $this->feed = new $class($this->model, $strict);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Proxy to allow feed element names to be used as method names
 | |
|      *
 | |
|      * For top-level feed elements we will provide access using methods or 
 | |
|      * attributes. This function simply passes on a request to the appropriate 
 | |
|      * feed type object.
 | |
|      *
 | |
|      * @param   string  $call - the method being called
 | |
|      * @param   array   $attributes
 | |
|      */
 | |
|     function __call($call, $attributes)
 | |
|     {
 | |
|         $attributes = array_pad($attributes, 5, false);
 | |
|         list($a, $b, $c, $d, $e) = $attributes;
 | |
|         return $this->feed->$call($a, $b, $c, $d, $e);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Proxy to allow feed element names to be used as attribute names
 | |
|      *
 | |
|      * To allow variable-like access to feed-level data we use this
 | |
|      * method. It simply passes along to __call() which in turn passes
 | |
|      * along to the relevant object.
 | |
|      *
 | |
|      * @param   string  $val - the name of the variable required
 | |
|      */
 | |
|     function __get($val)
 | |
|     {
 | |
|         return $this->feed->$val;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Provides iteration functionality.
 | |
|      *
 | |
|      * Of course we must be able to iterate... This function simply increases
 | |
|      * our internal counter.
 | |
|      */
 | |
|     function next()
 | |
|     {
 | |
|         if (isset($this->current_item) && 
 | |
|             $this->current_item <= $this->feed->numberEntries - 1) {
 | |
|             ++$this->current_item;
 | |
|         } else if (! isset($this->current_item)) {
 | |
|             $this->current_item = 0;
 | |
|         } else {
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return XML_Feed_Type object for current element
 | |
|      *
 | |
|      * @return    XML_Feed_Parser_Type Object
 | |
|      */
 | |
|     function current()
 | |
|     {
 | |
|         return $this->getEntryByOffset($this->current_item);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * For iteration -- returns the key for the current stage in the array.
 | |
|      *
 | |
|      * @return    int
 | |
|      */    
 | |
|     function key()
 | |
|     {
 | |
|         return $this->current_item;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * For iteration -- tells whether we have reached the 
 | |
|      * end.
 | |
|      *
 | |
|      * @return    bool
 | |
|      */
 | |
|     function valid()
 | |
|     {
 | |
|         return $this->current_item < $this->feed->numberEntries;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * For iteration -- resets the internal counter to the beginning.
 | |
|      */
 | |
|     function rewind()
 | |
|     {
 | |
|         $this->current_item = 0;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Provides access to entries by ID if one is specified in the source feed.
 | |
|      *
 | |
|      * As well as allowing the items to be iterated over we want to allow
 | |
|      * users to be able to access a specific entry. This is one of two ways of
 | |
|      * doing that, the other being by offset. This method can be quite slow
 | |
|      * if dealing with a large feed that hasn't yet been processed as it
 | |
|      * instantiates objects for every entry until it finds the one needed.
 | |
|      *
 | |
|      * @param    string    $id  Valid ID for the given feed format
 | |
|      * @return    XML_Feed_Parser_Type|false
 | |
|      */            
 | |
|     function getEntryById($id)
 | |
|     {
 | |
|         if (isset($this->idMappings[$id])) {
 | |
|             return $this->getEntryByOffset($this->idMappings[$id]);
 | |
|         }
 | |
| 
 | |
|         /* 
 | |
|          * Since we have not yet encountered that ID, let's go through all the
 | |
|          * remaining entries in order till we find it.
 | |
|          * This is a fairly slow implementation, but it should work.
 | |
|          */
 | |
|         return $this->feed->getEntryById($id);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Retrieve entry by numeric offset, starting from zero.
 | |
|      *
 | |
|      * As well as allowing the items to be iterated over we want to allow
 | |
|      * users to be able to access a specific entry. This is one of two ways of
 | |
|      * doing that, the other being by ID.
 | |
|      *
 | |
|      * @param    int    $offset The position of the entry within the feed, starting from 0
 | |
|      * @return    XML_Feed_Parser_Type|false
 | |
|      */
 | |
|     function getEntryByOffset($offset)
 | |
|     {
 | |
|         if ($offset < $this->feed->numberEntries) {
 | |
|             if (isset($this->feed->entries[$offset])) {
 | |
|                 return $this->feed->entries[$offset];
 | |
|             } else {
 | |
|                 try {
 | |
|                     $this->feed->getEntryByOffset($offset);
 | |
|                 } catch (Exception $e) {
 | |
|                     return false;
 | |
|                 }
 | |
|                 $id = $this->feed->entries[$offset]->getID();
 | |
|                 $this->idMappings[$id] = $offset;
 | |
|                 return $this->feed->entries[$offset];
 | |
|             }
 | |
|         } else {
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Retrieve version details from feed type class.
 | |
|      *
 | |
|      * @return void
 | |
|      * @author James Stewart
 | |
|      */
 | |
|     function version()
 | |
|     {
 | |
|         return $this->feed->version;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Returns a string representation of the feed.
 | |
|      * 
 | |
|      * @return String
 | |
|      **/
 | |
|     function __toString()
 | |
|     {
 | |
|         return $this->feed->__toString();
 | |
|     }
 | |
| }
 | |
| ?>
 |