forked from GNUsocial/gnu-social
		
	
		
			
	
	
		
			351 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			351 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
|   | <?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(); | ||
|  |     } | ||
|  | } | ||
|  | ?>
 |