| 
									
										
										
										
											2010-02-12 00:42:42 -05:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * StatusNet, the distributed open-source microblogging tool | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * An activity | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * PHP version 5 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * LICENCE: This program is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU Affero General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU Affero General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Affero General Public License | 
					
						
							|  |  |  |  * along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-02-22 17:10:50 -08:00
										 |  |  |  * @category  Feed | 
					
						
							| 
									
										
										
										
											2010-02-12 00:42:42 -05:00
										 |  |  |  * @package   StatusNet | 
					
						
							|  |  |  |  * @author    Evan Prodromou <evan@status.net> | 
					
						
							| 
									
										
										
										
											2010-02-22 17:10:50 -08:00
										 |  |  |  * @author    Zach Copley <zach@status.net> | 
					
						
							| 
									
										
										
										
											2010-02-12 00:42:42 -05:00
										 |  |  |  * @copyright 2010 StatusNet, Inc. | 
					
						
							|  |  |  |  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html AGPLv3 | 
					
						
							|  |  |  |  * @link      http://status.net/ | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if (!defined('STATUSNET')) { | 
					
						
							|  |  |  |     exit(1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  | /** | 
					
						
							|  |  |  |  * An activity in the ActivityStrea.ms world | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * An activity is kind of like a sentence: someone did something | 
					
						
							|  |  |  |  * to something else. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 'someone' is the 'actor'; 'did something' is the verb; | 
					
						
							|  |  |  |  * 'something else' is the object. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @category  OStatus | 
					
						
							|  |  |  |  * @package   StatusNet | 
					
						
							|  |  |  |  * @author    Evan Prodromou <evan@status.net> | 
					
						
							|  |  |  |  * @copyright 2010 StatusNet, Inc. | 
					
						
							|  |  |  |  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html AGPLv3 | 
					
						
							|  |  |  |  * @link      http://status.net/ | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class Activity | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const SPEC   = 'http://activitystrea.ms/spec/1.0/'; | 
					
						
							|  |  |  |     const SCHEMA = 'http://activitystrea.ms/schema/1.0/'; | 
					
						
							| 
									
										
										
										
											2010-03-22 18:53:09 -07:00
										 |  |  |     const MEDIA  = 'http://purl.org/syndication/atommedia'; | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const VERB       = 'verb'; | 
					
						
							|  |  |  |     const OBJECT     = 'object'; | 
					
						
							|  |  |  |     const ACTOR      = 'actor'; | 
					
						
							|  |  |  |     const SUBJECT    = 'subject'; | 
					
						
							|  |  |  |     const OBJECTTYPE = 'object-type'; | 
					
						
							|  |  |  |     const CONTEXT    = 'context'; | 
					
						
							|  |  |  |     const TARGET     = 'target'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const ATOM = 'http://www.w3.org/2005/Atom'; | 
					
						
							| 
									
										
										
										
											2010-02-12 00:42:42 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |     const AUTHOR    = 'author'; | 
					
						
							|  |  |  |     const PUBLISHED = 'published'; | 
					
						
							| 
									
										
										
										
											2010-02-14 13:19:32 -05:00
										 |  |  |     const UPDATED   = 'updated'; | 
					
						
							| 
									
										
										
										
											2010-02-12 00:42:42 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |     const RSS = null; // no namespace!
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const PUBDATE     = 'pubDate'; | 
					
						
							|  |  |  |     const DESCRIPTION = 'description'; | 
					
						
							|  |  |  |     const GUID        = 'guid'; | 
					
						
							|  |  |  |     const SELF        = 'self'; | 
					
						
							|  |  |  |     const IMAGE       = 'image'; | 
					
						
							|  |  |  |     const URL         = 'url'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const DC = 'http://purl.org/dc/elements/1.1/'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const CREATOR = 'creator'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const CONTENTNS = 'http://purl.org/rss/1.0/modules/content/'; | 
					
						
							| 
									
										
										
										
											2010-04-23 15:40:48 -07:00
										 |  |  |     const ENCODED = 'encoded'; | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |     public $actor;   // an ActivityObject
 | 
					
						
							|  |  |  |     public $verb;    // a string (the URL)
 | 
					
						
							| 
									
										
										
										
											2010-03-22 18:53:09 -07:00
										 |  |  |     public $objects = array();  // an array of ActivityObjects
 | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |     public $target;  // an ActivityObject
 | 
					
						
							|  |  |  |     public $context; // an ActivityObject
 | 
					
						
							|  |  |  |     public $time;    // Time of the activity
 | 
					
						
							|  |  |  |     public $link;    // an ActivityObject
 | 
					
						
							|  |  |  |     public $entry;   // the source entry
 | 
					
						
							|  |  |  |     public $feed;    // the source feed
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  |     public $summary; // summary of activity
 | 
					
						
							|  |  |  |     public $content; // HTML content of activity
 | 
					
						
							|  |  |  |     public $id;      // ID of the activity
 | 
					
						
							|  |  |  |     public $title;   // title of the activity
 | 
					
						
							| 
									
										
										
										
											2010-02-25 11:26:33 -08:00
										 |  |  |     public $categories = array(); // list of AtomCategory objects
 | 
					
						
							| 
									
										
										
										
											2010-03-02 16:30:09 -08:00
										 |  |  |     public $enclosures = array(); // list of enclosure URL references
 | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |     public $extra = array(); // extra elements as array(tag, attrs, content)
 | 
					
						
							|  |  |  |     public $source;  // ActivitySource object representing 'home feed'
 | 
					
						
							|  |  |  |     public $selfLink; // <link rel='self' type='application/atom+xml'>
 | 
					
						
							|  |  |  |     public $editLink; // <link rel='edit' type='application/atom+xml'>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Turns a regular old Atom <entry> into a magical activity | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param DOMElement $entry Atom entry to poke at | 
					
						
							|  |  |  |      * @param DOMElement $feed  Atom feed, for context | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  |     function __construct($entry = null, $feed = null) | 
					
						
							| 
									
										
										
										
											2010-02-12 00:42:42 -05:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  |         if (is_null($entry)) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-15 19:06:06 -07:00
										 |  |  |         // Insist on a feed's root DOMElement; don't allow a DOMDocument
 | 
					
						
							| 
									
										
										
										
											2010-03-03 20:55:53 -08:00
										 |  |  |         if ($feed instanceof DOMDocument) { | 
					
						
							| 
									
										
										
										
											2010-03-15 19:06:06 -07:00
										 |  |  |             throw new ClientException( | 
					
						
							| 
									
										
										
										
											2010-04-10 22:50:15 +02:00
										 |  |  |                 // TRANS: Client exception thrown when a feed instance is a DOMDocument.
 | 
					
						
							|  |  |  |                 _('Expecting a root feed element but got a whole XML document.') | 
					
						
							| 
									
										
										
										
											2010-03-03 20:55:53 -08:00
										 |  |  |             ); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |         $this->entry = $entry; | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |         $this->feed  = $feed; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |         if ($entry->namespaceURI == Activity::ATOM && | 
					
						
							|  |  |  |             $entry->localName == 'entry') { | 
					
						
							|  |  |  |             $this->_fromAtomEntry($entry, $feed); | 
					
						
							|  |  |  |         } else if ($entry->namespaceURI == Activity::RSS && | 
					
						
							|  |  |  |                    $entry->localName == 'item') { | 
					
						
							|  |  |  |             $this->_fromRssItem($entry, $feed); | 
					
						
							| 
									
										
										
										
											2011-07-19 16:02:28 -04:00
										 |  |  |         } else if ($entry->namespaceURI == Activity::SPEC && | 
					
						
							|  |  |  |                    $entry->localName == 'object') { | 
					
						
							|  |  |  |             $this->_fromAtomEntry($entry, $feed); | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-09-29 00:39:45 +02:00
										 |  |  |             // Low level exception. No need for i18n.
 | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |             throw new Exception("Unknown DOM element: {$entry->namespaceURI} {$entry->localName}"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function _fromAtomEntry($entry, $feed) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |         $pubEl = $this->_child($entry, self::PUBLISHED, self::ATOM); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($pubEl)) { | 
					
						
							|  |  |  |             $this->time = strtotime($pubEl->textContent); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             // XXX technically an error; being liberal. Good idea...?
 | 
					
						
							| 
									
										
										
										
											2010-02-14 13:19:32 -05:00
										 |  |  |             $updateEl = $this->_child($entry, self::UPDATED, self::ATOM); | 
					
						
							|  |  |  |             if (!empty($updateEl)) { | 
					
						
							|  |  |  |                 $this->time = strtotime($updateEl->textContent); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 $this->time = null; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-16 12:49:54 -05:00
										 |  |  |         $this->link = ActivityUtils::getPermalink($entry); | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $verbEl = $this->_child($entry, self::VERB); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($verbEl)) { | 
					
						
							|  |  |  |             $this->verb = trim($verbEl->textContent); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $this->verb = ActivityVerb::POST; | 
					
						
							|  |  |  |             // XXX: do other implied stuff here
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-19 16:38:58 -04:00
										 |  |  |         // get immediate object children
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $objectEls = ActivityUtils::children($entry, self::OBJECT, self::SPEC); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (count($objectEls) > 0) { | 
					
						
							|  |  |  |             foreach ($objectEls as $objectEl) { | 
					
						
							| 
									
										
										
										
											2011-07-19 16:02:28 -04:00
										 |  |  |                 // Special case for embedded activities
 | 
					
						
							|  |  |  |                 $objectType = ActivityUtils::childContent($objectEl, self::OBJECTTYPE, self::SPEC); | 
					
						
							|  |  |  |                 if (!empty($objectType) && $objectType == ActivityObject::ACTIVITY) { | 
					
						
							|  |  |  |                     $this->objects[] = new Activity($objectEl); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     $this->objects[] = new ActivityObject($objectEl); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-03-22 18:53:09 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2011-07-19 16:02:28 -04:00
										 |  |  |             // XXX: really?
 | 
					
						
							| 
									
										
										
										
											2010-03-22 18:53:09 -07:00
										 |  |  |             $this->objects[] = new ActivityObject($entry); | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $actorEl = $this->_child($entry, self::ACTOR); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($actorEl)) { | 
					
						
							| 
									
										
										
										
											2011-01-06 15:05:29 -08:00
										 |  |  |             // Standalone <activity:actor> elements are a holdover from older
 | 
					
						
							|  |  |  |             // versions of ActivityStreams. Newer feeds should have this data
 | 
					
						
							|  |  |  |             // integrated straight into <atom:author>.
 | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |             $this->actor = new ActivityObject($actorEl); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-27 22:44:10 -04:00
										 |  |  |             // Cliqset has bad actor IDs (just nickname of user). We
 | 
					
						
							|  |  |  |             // work around it by getting the author data and using its
 | 
					
						
							|  |  |  |             // id instead
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!preg_match('/^\w+:/', $this->actor->id)) { | 
					
						
							|  |  |  |                 $authorEl = ActivityUtils::child($entry, 'author'); | 
					
						
							|  |  |  |                 if (!empty($authorEl)) { | 
					
						
							|  |  |  |                     $authorObj = new ActivityObject($authorEl); | 
					
						
							|  |  |  |                     $this->actor->id = $authorObj->id; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |         } else if ($authorEl = $this->_child($entry, self::AUTHOR, self::ATOM)) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-06 15:05:29 -08:00
										 |  |  |             // An <atom:author> in the entry overrides any author info on
 | 
					
						
							|  |  |  |             // the surrounding feed.
 | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |             $this->actor = new ActivityObject($authorEl); | 
					
						
							| 
									
										
										
										
											2010-02-14 13:19:32 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-06 15:05:29 -08:00
										 |  |  |         } else if (!empty($feed) && | 
					
						
							|  |  |  |                    $subjectEl = $this->_child($feed, self::SUBJECT)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Feed subject is used for things like groups.
 | 
					
						
							|  |  |  |             // Should actually possibly not be interpreted as an actor...?
 | 
					
						
							|  |  |  |             $this->actor = new ActivityObject($subjectEl); | 
					
						
							| 
									
										
										
										
											2011-02-11 03:01:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else if (!empty($feed) && $authorEl = $this->_child($feed, self::AUTHOR, | 
					
						
							|  |  |  |                                                               self::ATOM)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // If there's no <atom:author> on the entry, it's safe to assume
 | 
					
						
							|  |  |  |             // the containing feed's authorship info applies.
 | 
					
						
							|  |  |  |             $this->actor = new ActivityObject($authorEl); | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $contextEl = $this->_child($entry, self::CONTEXT); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($contextEl)) { | 
					
						
							| 
									
										
										
										
											2010-02-18 22:18:14 -05:00
										 |  |  |             $this->context = new ActivityContext($contextEl); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $this->context = new ActivityContext($entry); | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $targetEl = $this->_child($entry, self::TARGET); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($targetEl)) { | 
					
						
							|  |  |  |             $this->target = new ActivityObject($targetEl); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $this->summary = ActivityUtils::childContent($entry, 'summary'); | 
					
						
							|  |  |  |         $this->id      = ActivityUtils::childContent($entry, 'id'); | 
					
						
							|  |  |  |         $this->content = ActivityUtils::getContent($entry); | 
					
						
							| 
									
										
										
										
											2010-02-25 11:26:33 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $catEls = $entry->getElementsByTagNameNS(self::ATOM, 'category'); | 
					
						
							|  |  |  |         if ($catEls) { | 
					
						
							|  |  |  |             for ($i = 0; $i < $catEls->length; $i++) { | 
					
						
							|  |  |  |                 $catEl = $catEls->item($i); | 
					
						
							|  |  |  |                 $this->categories[] = new AtomCategory($catEl); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-03-02 16:30:09 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         foreach (ActivityUtils::getLinks($entry, 'enclosure') as $link) { | 
					
						
							|  |  |  |             $this->enclosures[] = $link->getAttribute('href'); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // From APP. Might be useful.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->selfLink = ActivityUtils::getLink($entry, 'self', 'application/atom+xml'); | 
					
						
							|  |  |  |         $this->editLink = ActivityUtils::getLink($entry, 'edit', 'application/atom+xml'); | 
					
						
							| 
									
										
										
										
											2010-02-12 00:42:42 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-20 07:19:54 -05:00
										 |  |  |     function _fromRssItem($item, $channel) | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         $verbEl = $this->_child($item, self::VERB); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($verbEl)) { | 
					
						
							|  |  |  |             $this->verb = trim($verbEl->textContent); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $this->verb = ActivityVerb::POST; | 
					
						
							|  |  |  |             // XXX: do other implied stuff here
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $pubDateEl = $this->_child($item, self::PUBDATE, self::RSS); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($pubDateEl)) { | 
					
						
							|  |  |  |             $this->time = strtotime($pubDateEl->textContent); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-20 16:53:30 -05:00
										 |  |  |         if ($authorEl = $this->_child($item, self::AUTHOR, self::RSS)) { | 
					
						
							| 
									
										
										
										
											2010-03-19 15:41:48 -05:00
										 |  |  |             $this->actor = ActivityObject::fromRssAuthor($authorEl); | 
					
						
							| 
									
										
										
										
											2010-03-20 16:53:30 -05:00
										 |  |  |         } else if ($dcCreatorEl = $this->_child($item, self::CREATOR, self::DC)) { | 
					
						
							|  |  |  |             $this->actor = ActivityObject::fromDcCreator($dcCreatorEl); | 
					
						
							|  |  |  |         } else if ($posterousEl = $this->_child($item, ActivityObject::AUTHOR, ActivityObject::POSTEROUS)) { | 
					
						
							|  |  |  |             // Special case for Posterous.com
 | 
					
						
							|  |  |  |             $this->actor = ActivityObject::fromPosterousAuthor($posterousEl); | 
					
						
							|  |  |  |         } else if (!empty($channel)) { | 
					
						
							|  |  |  |             $this->actor = ActivityObject::fromRssChannel($channel); | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-03-20 16:53:30 -05:00
										 |  |  |             // No actor!
 | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->title = ActivityUtils::childContent($item, ActivityObject::TITLE, self::RSS); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-23 15:40:48 -07:00
										 |  |  |         $contentEl = ActivityUtils::child($item, self::ENCODED, self::CONTENTNS); | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($contentEl)) { | 
					
						
							| 
									
										
										
										
											2010-04-23 15:40:48 -07:00
										 |  |  |             // <content:encoded> XML node's text content is HTML; no further processing needed.
 | 
					
						
							|  |  |  |             $this->content = $contentEl->textContent; | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             $descriptionEl = ActivityUtils::child($item, self::DESCRIPTION, self::RSS); | 
					
						
							|  |  |  |             if (!empty($descriptionEl)) { | 
					
						
							| 
									
										
										
										
											2010-04-23 15:40:48 -07:00
										 |  |  |                 // Per spec, <description> must be plaintext.
 | 
					
						
							|  |  |  |                 // In practice, often there's HTML... but these days good
 | 
					
						
							|  |  |  |                 // feeds are using <content:encoded> which is explicitly
 | 
					
						
							|  |  |  |                 // real HTML.
 | 
					
						
							|  |  |  |                 // We'll treat this following spec, and do HTML escaping
 | 
					
						
							|  |  |  |                 // to convert from plaintext to HTML.
 | 
					
						
							|  |  |  |                 $this->content = htmlspecialchars($descriptionEl->textContent); | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->link = ActivityUtils::childContent($item, ActivityUtils::LINK, self::RSS); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // @fixme enclosures
 | 
					
						
							|  |  |  |         // @fixme thumbnails... maybe
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $guidEl = ActivityUtils::child($item, self::GUID, self::RSS); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($guidEl)) { | 
					
						
							|  |  |  |             $this->id = $guidEl->textContent; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if ($guidEl->hasAttribute('isPermaLink') && $guidEl->getAttribute('isPermaLink') != 'false') { | 
					
						
							|  |  |  |                 // overwrites <link>
 | 
					
						
							|  |  |  |                 $this->link = $this->id; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-22 18:53:09 -07:00
										 |  |  |         $this->objects[] = new ActivityObject($item); | 
					
						
							|  |  |  |         $this->context   = new ActivityContext($item); | 
					
						
							| 
									
										
										
										
											2010-03-19 09:48:39 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Returns an Atom <entry> based on this activity | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return DOMElement Atom entry | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2011-01-02 15:21:56 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-12 00:42:42 -05:00
										 |  |  |     function toAtomEntry() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |         return null; | 
					
						
							| 
									
										
										
										
											2010-02-12 00:42:42 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-15 20:25:39 -08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Returns an array based on this activity suitable | 
					
						
							|  |  |  |      * for encoding as a JSON object | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array $activity | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function asArray() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $activity = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // actor
 | 
					
						
							|  |  |  |         $activity['actor'] = $this->actor->asArray(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // body
 | 
					
						
							|  |  |  |         $activity['body'] = $this->content; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-18 18:10:21 -08:00
										 |  |  |         // generator <-- We could use this when we know a notice is created
 | 
					
						
							|  |  |  |         //               locally. Or if we know the upstream Generator.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // icon <-- I've decided to use the posting user's stream avatar here
 | 
					
						
							|  |  |  |         //          for now (also included in the avatarLinks extension)
 | 
					
						
							| 
									
										
										
										
											2011-02-15 20:25:39 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // object
 | 
					
						
							|  |  |  |         if ($this->verb == ActivityVerb::POST && count($this->objects) == 1) { | 
					
						
							|  |  |  |             $activity['object'] = $this->objects[0]->asArray(); | 
					
						
							| 
									
										
										
										
											2011-02-16 18:14:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-18 10:02:41 -08:00
										 |  |  |             // Context stuff. For now I'm just sticking most of it
 | 
					
						
							|  |  |  |             // in a property called "context"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($this->context)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (!empty($this->context->location)) { | 
					
						
							|  |  |  |                     $loc = $this->context->location; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     // GeoJSON
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     $activity['geopoint'] = array( | 
					
						
							|  |  |  |                         'type'        => 'Point', | 
					
						
							|  |  |  |                         'coordinates' => array($loc->lat, $loc->lon) | 
					
						
							|  |  |  |                     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 $activity['to']      = $this->context->getToArray(); | 
					
						
							|  |  |  |                 $activity['context'] = $this->context->asArray(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-16 18:14:58 -08:00
										 |  |  |             // Instead of adding enclosures as an extension to JSON
 | 
					
						
							|  |  |  |             // Activities, it seems like we should be using the
 | 
					
						
							|  |  |  |             // attachedObjects property of ActivityObject
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $attachedObjects = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // XXX: OK, this is kinda cheating. We should probably figure out
 | 
					
						
							|  |  |  |             // what kind of objects these are based on mime-type and then
 | 
					
						
							|  |  |  |             // create specific object types. Right now this rely on
 | 
					
						
							|  |  |  |             // duck-typing.  Also, we should include an embed code for
 | 
					
						
							|  |  |  |             // video attachments.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             foreach ($this->enclosures as $enclosure) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (is_string($enclosure)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     $attachedObjects[]['id']  = $enclosure; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     $attachedObjects[]['id']  = $enclosure->url; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     $mediaLink = new ActivityStreamsMediaLink( | 
					
						
							|  |  |  |                         $enclosure->url, | 
					
						
							|  |  |  |                         null, | 
					
						
							|  |  |  |                         null, | 
					
						
							|  |  |  |                         $enclosure->mimetype | 
					
						
							|  |  |  |                         // XXX: Add 'size' as an extension to MediaLink?
 | 
					
						
							|  |  |  |                     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     $attachedObjects[]['mediaLink'] = $mediaLink->asArray(); // extension
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if ($enclosure->title) { | 
					
						
							|  |  |  |                         $attachedObjects[]['displayName'] = $enclosure->title; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($attachedObjects)) { | 
					
						
							|  |  |  |                 $activity['object']['attachedObjects'] = $attachedObjects; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-15 20:25:39 -08:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             $activity['object'] = array(); | 
					
						
							|  |  |  |             foreach($this->objects as $object) { | 
					
						
							| 
									
										
										
										
											2011-07-18 17:06:03 -04:00
										 |  |  |                 $oa = $object->asArray(); | 
					
						
							|  |  |  |                 if ($object instanceof Activity) { | 
					
						
							|  |  |  |                     // throw in a type
 | 
					
						
							|  |  |  |                     // XXX: hackety-hack
 | 
					
						
							| 
									
										
										
										
											2011-07-18 17:42:28 -04:00
										 |  |  |                     $oa['objectType'] = 'activity'; | 
					
						
							| 
									
										
										
										
											2011-07-18 17:06:03 -04:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 $activity['object'][] = $oa; | 
					
						
							| 
									
										
										
										
											2011-02-15 20:25:39 -08:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $activity['postedTime'] = self::iso8601Date($this->time); // Change to exactly be RFC3339?
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-18 18:10:21 -08:00
										 |  |  |         // provider
 | 
					
						
							|  |  |  |         $provider = array( | 
					
						
							|  |  |  |             'objectType' => 'service', | 
					
						
							|  |  |  |             'displayName' => common_config('site', 'name'), | 
					
						
							|  |  |  |             'url' => common_root_url() | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $activity['provider'] = $provider; | 
					
						
							| 
									
										
										
										
											2011-02-15 20:25:39 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // target
 | 
					
						
							|  |  |  |         if (!empty($this->target)) { | 
					
						
							|  |  |  |             $activity['target'] = $this->target->asArray(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // title
 | 
					
						
							|  |  |  |         $activity['title'] = $this->title; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-16 18:14:58 -08:00
										 |  |  |         // updatedTime <-- Should we use this to indicate the time we received
 | 
					
						
							|  |  |  |         //                 a remote notice? Probably not.
 | 
					
						
							| 
									
										
										
										
											2011-02-15 20:25:39 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // verb
 | 
					
						
							| 
									
										
										
										
											2011-02-16 16:44:02 -08:00
										 |  |  |         //
 | 
					
						
							|  |  |  |         // We can probably use the whole schema URL here but probably the
 | 
					
						
							|  |  |  |         // relative simple name is easier to parse
 | 
					
						
							|  |  |  |         $activity['verb'] = substr($this->verb, strrpos($this->verb, '/') + 1); | 
					
						
							| 
									
										
										
										
											2011-02-15 20:25:39 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-16 18:14:58 -08:00
										 |  |  |         /* Purely extensions hereafter */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-18 09:29:49 -08:00
										 |  |  |         $tags = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Use an Activity Object for term? Which object? Note?
 | 
					
						
							|  |  |  |         foreach ($this->categories as $cat) { | 
					
						
							|  |  |  |             $tags[] = $cat->term; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $activity['tags'] = $tags; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-16 18:55:13 -08:00
										 |  |  |         // XXX: a bit of a hack... Since JSON isn't namespaced we probably
 | 
					
						
							|  |  |  |         // shouldn't be using 'statusnet:notice_info', but this will work
 | 
					
						
							|  |  |  |         // for the moment.
 | 
					
						
							| 
									
										
										
										
											2011-02-16 18:14:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-16 18:55:13 -08:00
										 |  |  |         foreach ($this->extra as $e) { | 
					
						
							|  |  |  |             list($objectName, $props, $txt) = $e; | 
					
						
							| 
									
										
										
										
											2011-02-17 19:02:57 -08:00
										 |  |  |             if (!empty($objectName)) { | 
					
						
							|  |  |  |                 $activity[$objectName] = $props; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-02-16 18:55:13 -08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-02-17 20:12:28 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-16 16:21:46 -08:00
										 |  |  |         return array_filter($activity); | 
					
						
							| 
									
										
										
										
											2011-02-15 20:25:39 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-08 07:25:55 -05:00
										 |  |  |     function asString($namespace=false, $author=true, $source=false) | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         $xs = new XMLStringer(true); | 
					
						
							| 
									
										
										
										
											2011-01-02 15:21:56 -08:00
										 |  |  |         $this->outputTo($xs, $namespace, $author, $source); | 
					
						
							|  |  |  |         return $xs->getString(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-18 17:06:03 -04:00
										 |  |  |     function outputTo($xs, $namespace=false, $author=true, $source=false, $tag='entry') | 
					
						
							| 
									
										
										
										
											2011-01-02 15:21:56 -08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  |         if ($namespace) { | 
					
						
							|  |  |  |             $attrs = array('xmlns' => 'http://www.w3.org/2005/Atom', | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |                            'xmlns:thr' => 'http://purl.org/syndication/thread/1.0', | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  |                            'xmlns:activity' => 'http://activitystrea.ms/spec/1.0/', | 
					
						
							| 
									
										
										
										
											2010-02-23 20:44:27 +00:00
										 |  |  |                            'xmlns:georss' => 'http://www.georss.org/georss', | 
					
						
							|  |  |  |                            'xmlns:ostatus' => 'http://ostatus.org/schema/1.0', | 
					
						
							| 
									
										
										
										
											2010-02-25 16:06:49 -08:00
										 |  |  |                            'xmlns:poco' => 'http://portablecontacts.net/spec/1.0', | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |                            'xmlns:media' => 'http://purl.org/syndication/atommedia', | 
					
						
							|  |  |  |                            'xmlns:statusnet' => 'http://status.net/schema/api/1/'); | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             $attrs = array(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-18 17:06:03 -04:00
										 |  |  |         $xs->elementStart($tag, $attrs); | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-18 17:06:03 -04:00
										 |  |  |         if ($tag != 'entry') { | 
					
						
							|  |  |  |             $xs->element('activity:object-type', null, ActivityObject::ACTIVITY); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($this->verb == ActivityVerb::POST && count($this->objects) == 1 && $tag == 'entry') { | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |             $obj = $this->objects[0]; | 
					
						
							| 
									
										
										
										
											2010-12-20 13:26:57 -05:00
										 |  |  | 			$obj->outputTo($xs, null); | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $xs->element('id', null, $this->id); | 
					
						
							|  |  |  |             $xs->element('title', null, $this->title); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $xs->element('content', array('type' => 'html'), $this->content); | 
					
						
							| 
									
										
										
										
											2011-01-29 00:33:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |             if (!empty($this->summary)) { | 
					
						
							|  |  |  |                 $xs->element('summary', null, $this->summary); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-01-29 00:33:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |             if (!empty($this->link)) { | 
					
						
							|  |  |  |                 $xs->element('link', array('rel' => 'alternate', | 
					
						
							|  |  |  |                                            'type' => 'text/html'), | 
					
						
							|  |  |  |                              $this->link); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |         $xs->element('activity:verb', null, $this->verb); | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |         $published = self::iso8601Date($this->time); | 
					
						
							| 
									
										
										
										
											2011-01-29 00:33:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |         $xs->element('published', null, $published); | 
					
						
							|  |  |  |         $xs->element('updated', null, $published); | 
					
						
							| 
									
										
										
										
											2011-01-29 00:33:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-15 07:11:50 -04:00
										 |  |  |         if ($author) { | 
					
						
							| 
									
										
										
										
											2010-12-27 09:25:26 -08:00
										 |  |  |             $this->actor->outputTo($xs, 'author'); | 
					
						
							| 
									
										
										
										
											2011-02-09 23:18:14 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             // XXX: Remove <activity:actor> ASAP! Author information
 | 
					
						
							|  |  |  |             // has been moved to the author element in the Activity
 | 
					
						
							|  |  |  |             // Streams spec. We're outputting actor only for backward
 | 
					
						
							|  |  |  |             // compatibility with clients that can only parse
 | 
					
						
							|  |  |  |             // activities based on older versions of the spec.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $depMsg = 'Deprecation warning: activity:actor is present ' | 
					
						
							|  |  |  |                 . 'only for backward compatibility. It will be ' | 
					
						
							|  |  |  |                 . 'removed in the next version of StatusNet.'; | 
					
						
							|  |  |  |             $xs->comment($depMsg); | 
					
						
							|  |  |  |             $this->actor->outputTo($xs, 'activity:actor'); | 
					
						
							| 
									
										
										
										
											2010-02-24 17:36:31 +00:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-18 17:06:03 -04:00
										 |  |  |         if ($this->verb != ActivityVerb::POST || count($this->objects) != 1 || $tag != 'entry') { | 
					
						
							| 
									
										
										
										
											2010-03-22 18:53:09 -07:00
										 |  |  |             foreach($this->objects as $object) { | 
					
						
							| 
									
										
										
										
											2011-07-18 17:06:03 -04:00
										 |  |  |                 if ($object instanceof Activity) { | 
					
						
							|  |  |  |                     $object->outputTo($xs, false, true, true, 'activity:object'); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     $object->outputTo($xs, 'activity:object'); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-03-22 18:53:09 -07:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-24 17:36:31 +00:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |         if (!empty($this->context)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($this->context->replyToID)) { | 
					
						
							|  |  |  |                 if (!empty($this->context->replyToUrl)) { | 
					
						
							|  |  |  |                     $xs->element('thr:in-reply-to', | 
					
						
							|  |  |  |                                  array('ref' => $this->context->replyToID, | 
					
						
							|  |  |  |                                        'href' => $this->context->replyToUrl)); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     $xs->element('thr:in-reply-to', | 
					
						
							|  |  |  |                                  array('ref' => $this->context->replyToID)); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($this->context->replyToUrl)) { | 
					
						
							|  |  |  |                 $xs->element('link', array('rel' => 'related', | 
					
						
							|  |  |  |                                            'href' => $this->context->replyToUrl)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($this->context->conversation)) { | 
					
						
							|  |  |  |                 $xs->element('link', array('rel' => 'ostatus:conversation', | 
					
						
							|  |  |  |                                            'href' => $this->context->conversation)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             foreach ($this->context->attention as $attnURI) { | 
					
						
							|  |  |  |                 $xs->element('link', array('rel' => 'ostatus:attention', | 
					
						
							|  |  |  |                                            'href' => $attnURI)); | 
					
						
							|  |  |  |                 $xs->element('link', array('rel' => 'mentioned', | 
					
						
							|  |  |  |                                            'href' => $attnURI)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // XXX: shoulda used ActivityVerb::SHARE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($this->context->forwardID)) { | 
					
						
							|  |  |  |                 if (!empty($this->context->forwardUrl)) { | 
					
						
							|  |  |  |                     $xs->element('ostatus:forward', | 
					
						
							|  |  |  |                                  array('ref' => $this->context->forwardID, | 
					
						
							|  |  |  |                                        'href' => $this->context->forwardUrl)); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     $xs->element('ostatus:forward', | 
					
						
							|  |  |  |                                  array('ref' => $this->context->forwardID)); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($this->context->location)) { | 
					
						
							|  |  |  |                 $loc = $this->context->location; | 
					
						
							|  |  |  |                 $xs->element('georss:point', null, $loc->lat . ' ' . $loc->lon); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-24 17:36:31 +00:00
										 |  |  |         if ($this->target) { | 
					
						
							| 
									
										
										
										
											2010-12-20 13:26:57 -05:00
										 |  |  |             $this->target->outputTo($xs, 'activity:target'); | 
					
						
							| 
									
										
										
										
											2010-02-24 17:36:31 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-25 11:26:33 -08:00
										 |  |  |         foreach ($this->categories as $cat) { | 
					
						
							| 
									
										
										
										
											2010-12-27 09:46:25 -08:00
										 |  |  |             $cat->outputTo($xs); | 
					
						
							| 
									
										
										
										
											2010-02-25 11:26:33 -08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |         // can be either URLs or enclosure objects
 | 
					
						
							| 
									
										
										
										
											2011-01-29 00:33:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |         foreach ($this->enclosures as $enclosure) { | 
					
						
							|  |  |  |             if (is_string($enclosure)) { | 
					
						
							| 
									
										
										
										
											2010-12-06 15:51:42 -05:00
										 |  |  |                 $xs->element('link', array('rel' => 'enclosure', | 
					
						
							|  |  |  |                                            'href' => $enclosure)); | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 $attributes = array('rel' => 'enclosure', | 
					
						
							|  |  |  |                                     'href' => $enclosure->url, | 
					
						
							|  |  |  |                                     'type' => $enclosure->mimetype, | 
					
						
							|  |  |  |                                     'length' => $enclosure->size); | 
					
						
							|  |  |  |                 if ($enclosure->title) { | 
					
						
							|  |  |  |                     $attributes['title'] = $enclosure->title; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-12-06 15:51:42 -05:00
										 |  |  |                 $xs->element('link', $attributes); | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Info on the source feed
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-08 07:25:55 -05:00
										 |  |  |         if ($source && !empty($this->source)) { | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |             $xs->elementStart('source'); | 
					
						
							| 
									
										
										
										
											2011-01-29 00:33:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |             $xs->element('id', null, $this->source->id); | 
					
						
							|  |  |  |             $xs->element('title', null, $this->source->title); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (array_key_exists('alternate', $this->source->links)) { | 
					
						
							|  |  |  |                 $xs->element('link', array('rel' => 'alternate', | 
					
						
							|  |  |  |                                            'type' => 'text/html', | 
					
						
							|  |  |  |                                            'href' => $this->source->links['alternate'])); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-01-29 00:33:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |             if (array_key_exists('self', $this->source->links)) { | 
					
						
							|  |  |  |                 $xs->element('link', array('rel' => 'self', | 
					
						
							|  |  |  |                                            'type' => 'application/atom+xml', | 
					
						
							|  |  |  |                                            'href' => $this->source->links['self'])); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (array_key_exists('license', $this->source->links)) { | 
					
						
							|  |  |  |                 $xs->element('link', array('rel' => 'license', | 
					
						
							|  |  |  |                                            'href' => $this->source->links['license'])); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($this->source->icon)) { | 
					
						
							|  |  |  |                 $xs->element('icon', null, $this->source->icon); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($this->source->updated)) { | 
					
						
							|  |  |  |                 $xs->element('updated', null, $this->source->updated); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2011-01-29 00:33:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |             $xs->elementEnd('source'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($this->selfLink)) { | 
					
						
							|  |  |  |             $xs->element('link', array('rel' => 'self', | 
					
						
							|  |  |  |                                        'type' => 'application/atom+xml', | 
					
						
							|  |  |  |                                        'href' => $this->selfLink)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!empty($this->editLink)) { | 
					
						
							|  |  |  |             $xs->element('link', array('rel' => 'edit', | 
					
						
							|  |  |  |                                        'type' => 'application/atom+xml', | 
					
						
							|  |  |  |                                        'href' => $this->editLink)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // For throwing in extra elements; used for statusnet:notice_info
 | 
					
						
							| 
									
										
										
										
											2011-01-29 00:33:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-05 16:15:05 -05:00
										 |  |  |         foreach ($this->extra as $el) { | 
					
						
							|  |  |  |             list($tag, $attrs, $content) = $el; | 
					
						
							|  |  |  |             $xs->element($tag, $attrs, $content); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-18 17:06:03 -04:00
										 |  |  |         $xs->elementEnd($tag); | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-02 15:21:56 -08:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2010-02-21 10:50:51 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |     private function _child($element, $tag, $namespace=self::SPEC) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-02-16 12:49:54 -05:00
										 |  |  |         return ActivityUtils::child($element, $tag, $namespace); | 
					
						
							| 
									
										
										
										
											2010-02-14 12:12:47 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-09-13 17:05:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-30 12:36:54 -07:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * For consistency, we'll always output UTC rather than local time. | 
					
						
							|  |  |  |      * Note that clients *should* accept any timezone we give them as long | 
					
						
							|  |  |  |      * as it's properly formatted. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param int $tm Unix timestamp | 
					
						
							|  |  |  |      * @return string | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2010-09-13 17:05:02 -04:00
										 |  |  |     static function iso8601Date($tm) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $dateStr = date('d F Y H:i:s', $tm); | 
					
						
							|  |  |  |         $d = new DateTime($dateStr, new DateTimeZone('UTC')); | 
					
						
							|  |  |  |         return $d->format('c'); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-25 11:26:33 -08:00
										 |  |  | } |