| <?php |
| |
| /** |
| * Zend Framework |
| * |
| * LICENSE |
| * |
| * This source file is subject to the new BSD license that is bundled |
| * with this package in the file LICENSE.txt. |
| * It is also available through the world-wide-web at this URL: |
| * http://framework.zend.com/license/new-bsd |
| * If you did not receive a copy of the license and are unable to |
| * obtain it through the world-wide-web, please send an email |
| * to license@zend.com so we can send you a copy immediately. |
| * |
| * @category Zend |
| * @package Zend_Feed |
| * @copyright Copyright (c) 2005-2008 Zend Technologies USA Inc. (http://www.zend.com) |
| * @license http://framework.zend.com/license/new-bsd New BSD License |
| * @version $Id: Builder.php 8064 2008-02-16 10:58:39Z thomas $ |
| */ |
| |
| /** |
| * @see Zend_Feed_Builder_Interface |
| */ |
| require_once 'external/Zend/Feed/Builder/Interface.php'; |
| |
| /** |
| * @see Zend_Feed_Builder_Header |
| */ |
| require_once 'external/Zend/Feed/Builder/Header.php'; |
| |
| /** |
| * @see Zend_Feed_Builder_Entry |
| */ |
| require_once 'external/Zend/Feed/Builder/Entry.php'; |
| |
| /** |
| * A simple implementation of Zend_Feed_Builder_Interface. |
| * |
| * Users are encouraged to make their own classes to implement Zend_Feed_Builder_Interface |
| * |
| * @category Zend |
| * @package Zend_Feed |
| * @copyright Copyright (c) 2005-2008 Zend Technologies USA Inc. (http://www.zend.com) |
| * @license http://framework.zend.com/license/new-bsd New BSD License |
| */ |
| class Zend_Feed_Builder implements Zend_Feed_Builder_Interface { |
| /** |
| * The data of the feed |
| * |
| * @var $_data array |
| */ |
| private $_data; |
| |
| /** |
| * Header of the feed |
| * |
| * @var $_header Zend_Feed_Builder_Header |
| */ |
| private $_header; |
| |
| /** |
| * List of the entries of the feed |
| * |
| * @var $_entries array |
| */ |
| private $_entries = array(); |
| |
| /** |
| * Constructor. The $data array must conform to the following format: |
| * <code> |
| * array( |
| * 'title' => 'title of the feed', //required |
| * 'link' => 'canonical url to the feed', //required |
| * 'lastUpdate' => 'timestamp of the update date', // optional |
| * 'published' => 'timestamp of the publication date', //optional |
| * 'charset' => 'charset', // required |
| * 'description' => 'short description of the feed', //optional |
| * 'author' => 'author/publisher of the feed', //optional |
| * 'email' => 'email of the author', //optional |
| * 'webmaster' => 'email address for person responsible for technical issues' // optional, ignored if atom is used |
| * 'copyright' => 'copyright notice', //optional |
| * 'image' => 'url to image', //optional |
| * 'generator' => 'generator', // optional |
| * 'language' => 'language the feed is written in', // optional |
| * 'ttl' => 'how long in minutes a feed can be cached before refreshing', // optional, ignored if atom is used |
| * 'rating' => 'The PICS rating for the channel.', // optional, ignored if atom is used |
| * 'cloud' => array( |
| * 'domain' => 'domain of the cloud, e.g. rpc.sys.com' // required |
| * 'port' => 'port to connect to' // optional, default to 80 |
| * 'path' => 'path of the cloud, e.g. /RPC2 //required |
| * 'registerProcedure' => 'procedure to call, e.g. myCloud.rssPleaseNotify' // required |
| * 'protocol' => 'protocol to use, e.g. soap or xml-rpc' // required |
| * ), a cloud to be notified of updates // optional, ignored if atom is used |
| * 'textInput' => array( |
| * 'title' => 'the label of the Submit button in the text input area' // required, |
| * 'description' => 'explains the text input area' // required |
| * 'name' => 'the name of the text object in the text input area' // required |
| * 'link' => 'the URL of the CGI script that processes text input requests' // required |
| * ) // a text input box that can be displayed with the feed // optional, ignored if atom is used |
| * 'skipHours' => array( |
| * 'hour in 24 format', // e.g 13 (1pm) |
| * // up to 24 rows whose value is a number between 0 and 23 |
| * ) // Hint telling aggregators which hours they can skip // optional, ignored if atom is used |
| * 'skipDays ' => array( |
| * 'a day to skip', // e.g Monday |
| * // up to 7 rows whose value is a Monday, Tuesday, Wednesday, Thursday, Friday, Saturday or Sunday |
| * ) // Hint telling aggregators which days they can skip // optional, ignored if atom is used |
| * 'itunes' => array( |
| * 'author' => 'Artist column' // optional, default to the main author value |
| * 'owner' => array( |
| * 'name' => 'name of the owner' // optional, default to main author value |
| * 'email' => 'email of the owner' // optional, default to main email value |
| * ) // Owner of the podcast // optional |
| * 'image' => 'album/podcast art' // optional, default to the main image value |
| * 'subtitle' => 'short description' // optional, default to the main description value |
| * 'summary' => 'longer description' // optional, default to the main description value |
| * 'block' => 'Prevent an episode from appearing (yes|no)' // optional |
| * 'category' => array( |
| * array('main' => 'main category', // required |
| * 'sub' => 'sub category' // optional |
| * ), |
| * // up to 3 rows |
| * ) // 'Category column and in iTunes Music Store Browse' // required |
| * 'explicit' => 'parental advisory graphic (yes|no|clean)' // optional |
| * 'keywords' => 'a comma separated list of 12 keywords maximum' // optional |
| * 'new-feed-url' => 'used to inform iTunes of new feed URL location' // optional |
| * ) // Itunes extension data // optional, ignored if atom is used |
| * 'entries' => array( |
| * array( |
| * 'title' => 'title of the feed entry', //required |
| * 'link' => 'url to a feed entry', //required |
| * 'description' => 'short version of a feed entry', // only text, no html, required |
| * 'guid' => 'id of the article, if not given link value will used', //optional |
| * 'content' => 'long version', // can contain html, optional |
| * 'lastUpdate' => 'timestamp of the publication date', // optional |
| * 'comments' => 'comments page of the feed entry', // optional |
| * 'commentRss' => 'the feed url of the associated comments', // optional |
| * 'source' => array( |
| * 'title' => 'title of the original source' // required, |
| * 'url' => 'url of the original source' // required |
| * ) // original source of the feed entry // optional |
| * 'category' => array( |
| * array( |
| * 'term' => 'first category label' // required, |
| * 'scheme' => 'url that identifies a categorization scheme' // optional |
| * ), |
| * array( |
| * //data for the second category and so on |
| * ) |
| * ) // list of the attached categories // optional |
| * 'enclosure' => array( |
| * array( |
| * 'url' => 'url of the linked enclosure' // required |
| * 'type' => 'mime type of the enclosure' // optional |
| * 'length' => 'length of the linked content in octets' // optional |
| * ), |
| * array( |
| * //data for the second enclosure and so on |
| * ) |
| * ) // list of the enclosures of the feed entry // optional |
| * ), |
| * array( |
| * //data for the second entry and so on |
| * ) |
| * ) |
| * ); |
| * </code> |
| * |
| * @param array $data |
| * @return void |
| */ |
| public function __construct(array $data) { |
| $this->_data = $data; |
| $this->_createHeader($data); |
| if (isset($data['entries'])) { |
| $this->_createEntries($data['entries']); |
| } |
| } |
| |
| /** |
| * Returns an instance of Zend_Feed_Builder_Header |
| * describing the header of the feed |
| * |
| * @return Zend_Feed_Builder_Header |
| */ |
| public function getHeader() { |
| return $this->_header; |
| } |
| |
| /** |
| * Returns an array of Zend_Feed_Builder_Entry instances |
| * describing the entries of the feed |
| * |
| * @return array of Zend_Feed_Builder_Entry |
| */ |
| public function getEntries() { |
| return $this->_entries; |
| } |
| |
| /** |
| * Create the Zend_Feed_Builder_Header instance |
| * |
| * @param array $data |
| * @throws Zend_Feed_Builder_Exception |
| * @return void |
| */ |
| private function _createHeader(array $data) { |
| $mandatories = array('title', 'link', 'charset'); |
| foreach ($mandatories as $mandatory) { |
| if (! isset($data[$mandatory])) { |
| /** |
| * @see Zend_Feed_Builder_Exception |
| */ |
| require_once 'external/Zend/Feed/Builder/Exception.php'; |
| throw new Zend_Feed_Builder_Exception("$mandatory key is missing"); |
| } |
| } |
| $this->_header = new Zend_Feed_Builder_Header($data['title'], $data['link'], $data['charset']); |
| if (isset($data['lastUpdate'])) { |
| $this->_header->setLastUpdate($data['lastUpdate']); |
| } |
| if (isset($data['published'])) { |
| $this->_header->setPublishedDate($data['published']); |
| } |
| if (isset($data['description'])) { |
| $this->_header->setDescription($data['description']); |
| } |
| if (isset($data['author'])) { |
| $this->_header->setAuthor($data['author']); |
| } |
| if (isset($data['email'])) { |
| $this->_header->setEmail($data['email']); |
| } |
| if (isset($data['webmaster'])) { |
| $this->_header->setWebmaster($data['webmaster']); |
| } |
| if (isset($data['copyright'])) { |
| $this->_header->setCopyright($data['copyright']); |
| } |
| if (isset($data['image'])) { |
| $this->_header->setImage($data['image']); |
| } |
| if (isset($data['generator'])) { |
| $this->_header->setGenerator($data['generator']); |
| } |
| if (isset($data['language'])) { |
| $this->_header->setLanguage($data['language']); |
| } |
| if (isset($data['ttl'])) { |
| $this->_header->setTtl($data['ttl']); |
| } |
| if (isset($data['rating'])) { |
| $this->_header->setRating($data['rating']); |
| } |
| if (isset($data['cloud'])) { |
| $mandatories = array('domain', 'path', 'registerProcedure', 'protocol'); |
| foreach ($mandatories as $mandatory) { |
| if (! isset($data['cloud'][$mandatory])) { |
| /** |
| * @see Zend_Feed_Builder_Exception |
| */ |
| require_once 'external/Zend/Feed/Builder/Exception.php'; |
| throw new Zend_Feed_Builder_Exception("you have to define $mandatory property of your cloud"); |
| } |
| } |
| $uri_str = 'http://' . $data['cloud']['domain'] . $data['cloud']['path']; |
| $this->_header->setCloud($uri_str, $data['cloud']['registerProcedure'], $data['cloud']['protocol']); |
| } |
| if (isset($data['textInput'])) { |
| $mandatories = array('title', 'description', 'name', 'link'); |
| foreach ($mandatories as $mandatory) { |
| if (! isset($data['textInput'][$mandatory])) { |
| /** |
| * @see Zend_Feed_Builder_Exception |
| */ |
| require_once 'external/Zend/Feed/Builder/Exception.php'; |
| throw new Zend_Feed_Builder_Exception("you have to define $mandatory property of your textInput"); |
| } |
| } |
| $this->_header->setTextInput($data['textInput']['title'], $data['textInput']['description'], $data['textInput']['name'], $data['textInput']['link']); |
| } |
| if (isset($data['skipHours'])) { |
| $this->_header->setSkipHours($data['skipHours']); |
| } |
| if (isset($data['skipDays'])) { |
| $this->_header->setSkipDays($data['skipDays']); |
| } |
| if (isset($data['itunes'])) { |
| $itunes = new Zend_Feed_Builder_Header_Itunes($data['itunes']['category']); |
| if (isset($data['itunes']['author'])) { |
| $itunes->setAuthor($data['itunes']['author']); |
| } |
| if (isset($data['itunes']['owner'])) { |
| $name = isset($data['itunes']['owner']['name']) ? $data['itunes']['owner']['name'] : ''; |
| $email = isset($data['itunes']['owner']['email']) ? $data['itunes']['owner']['email'] : ''; |
| $itunes->setOwner($name, $email); |
| } |
| if (isset($data['itunes']['image'])) { |
| $itunes->setImage($data['itunes']['image']); |
| } |
| if (isset($data['itunes']['subtitle'])) { |
| $itunes->setSubtitle($data['itunes']['subtitle']); |
| } |
| if (isset($data['itunes']['summary'])) { |
| $itunes->setSummary($data['itunes']['summary']); |
| } |
| if (isset($data['itunes']['block'])) { |
| $itunes->setBlock($data['itunes']['block']); |
| } |
| if (isset($data['itunes']['explicit'])) { |
| $itunes->setExplicit($data['itunes']['explicit']); |
| } |
| if (isset($data['itunes']['keywords'])) { |
| $itunes->setKeywords($data['itunes']['keywords']); |
| } |
| if (isset($data['itunes']['new-feed-url'])) { |
| $itunes->setNewFeedUrl($data['itunes']['new-feed-url']); |
| } |
| |
| $this->_header->setITunes($itunes); |
| } |
| } |
| |
| /** |
| * Create the array of article entries |
| * |
| * @param array $data |
| * @throws Zend_Feed_Builder_Exception |
| * @return void |
| */ |
| private function _createEntries(array $data) { |
| foreach ($data as $row) { |
| $mandatories = array('title', 'link', 'description'); |
| foreach ($mandatories as $mandatory) { |
| if (! isset($row[$mandatory])) { |
| /** |
| * @see Zend_Feed_Builder_Exception |
| */ |
| require_once 'external/Zend/Feed/Builder/Exception.php'; |
| throw new Zend_Feed_Builder_Exception("$mandatory key is missing"); |
| } |
| } |
| $entry = new Zend_Feed_Builder_Entry($row['title'], $row['link'], $row['description']); |
| if (isset($row['guid'])) { |
| $entry->setId($row['guid']); |
| } |
| if (isset($row['content'])) { |
| $entry->setContent($row['content']); |
| } |
| if (isset($row['lastUpdate'])) { |
| $entry->setLastUpdate($row['lastUpdate']); |
| } |
| if (isset($row['comments'])) { |
| $entry->setCommentsUrl($row['comments']); |
| } |
| if (isset($row['commentRss'])) { |
| $entry->setCommentsRssUrl($row['commentRss']); |
| } |
| if (isset($row['source'])) { |
| $mandatories = array('title', 'url'); |
| foreach ($mandatories as $mandatory) { |
| if (! isset($row['source'][$mandatory])) { |
| /** |
| * @see Zend_Feed_Builder_Exception |
| */ |
| require_once 'external/Zend/Feed/Builder/Exception.php'; |
| throw new Zend_Feed_Builder_Exception("$mandatory key of source property is missing"); |
| } |
| } |
| $entry->setSource($row['source']['title'], $row['source']['url']); |
| } |
| if (isset($row['category'])) { |
| $entry->setCategories($row['category']); |
| } |
| if (isset($row['enclosure'])) { |
| $entry->setEnclosures($row['enclosure']); |
| } |
| |
| $this->_entries[] = $entry; |
| } |
| } |
| } |