2010-03-20 21:06:22 +00: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/>.
|
|
|
|
*
|
|
|
|
* @category Feed
|
|
|
|
* @package StatusNet
|
|
|
|
* @author Evan Prodromou <evan@status.net>
|
|
|
|
* @author Zach Copley <zach@status.net>
|
|
|
|
* @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);
|
|
|
|
}
|
|
|
|
|
|
|
|
class ActivityContext
|
|
|
|
{
|
|
|
|
public $replyToID;
|
|
|
|
public $replyToUrl;
|
|
|
|
public $location;
|
2013-10-28 21:21:14 +00:00
|
|
|
public $attention = array(); // 'uri' => 'type'
|
2010-03-20 21:06:22 +00:00
|
|
|
public $conversation;
|
2017-05-02 17:58:22 +01:00
|
|
|
public $conversation_url;
|
2013-06-07 08:11:23 +01:00
|
|
|
public $scope;
|
2010-03-20 21:06:22 +00:00
|
|
|
|
|
|
|
const THR = 'http://purl.org/syndication/thread/1.0';
|
|
|
|
const GEORSS = 'http://www.georss.org/georss';
|
|
|
|
const OSTATUS = 'http://ostatus.org/schema/1.0';
|
|
|
|
|
2013-10-28 21:21:14 +00:00
|
|
|
const INREPLYTO = 'in-reply-to';
|
|
|
|
const REF = 'ref';
|
|
|
|
const HREF = 'href';
|
2014-11-27 12:39:38 +00:00
|
|
|
|
|
|
|
// OStatus element names with prefixes
|
2013-10-28 21:21:14 +00:00
|
|
|
const OBJECTTYPE = 'ostatus:object-type'; // FIXME: Undocumented!
|
2017-05-02 17:58:22 +01:00
|
|
|
const CONVERSATION = 'conversation';
|
2010-03-20 21:06:22 +00:00
|
|
|
|
|
|
|
const POINT = 'point';
|
|
|
|
|
2010-08-02 19:06:14 +01:00
|
|
|
const MENTIONED = 'mentioned';
|
2010-03-20 21:06:22 +00:00
|
|
|
|
2013-10-28 21:21:14 +00:00
|
|
|
const ATTN_PUBLIC = 'http://activityschema.org/collection/public';
|
|
|
|
|
2010-09-13 21:23:10 +01:00
|
|
|
function __construct($element = null)
|
2010-03-20 21:06:22 +00:00
|
|
|
{
|
2010-09-13 21:23:10 +01:00
|
|
|
if (empty($element)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-03-20 21:06:22 +00:00
|
|
|
$replyToEl = ActivityUtils::child($element, self::INREPLYTO, self::THR);
|
|
|
|
|
|
|
|
if (!empty($replyToEl)) {
|
|
|
|
$this->replyToID = $replyToEl->getAttribute(self::REF);
|
|
|
|
$this->replyToUrl = $replyToEl->getAttribute(self::HREF);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->location = $this->getLocation($element);
|
|
|
|
|
2017-05-02 17:58:22 +01:00
|
|
|
foreach ($element->getElementsByTagNameNS(self::OSTATUS, self::CONVERSATION) as $conv) {
|
|
|
|
if ($conv->hasAttribute('ref')) {
|
|
|
|
$this->conversation = $conv->getAttribute('ref');
|
|
|
|
if ($conv->hasAttribute('href')) {
|
|
|
|
$this->conversation_url = $conv->getAttribute('href');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$this->conversation = $conv->textContent;
|
|
|
|
}
|
|
|
|
if (!empty($this->conversation)) {
|
|
|
|
break;
|
|
|
|
}
|
2014-11-27 12:39:38 +00:00
|
|
|
}
|
|
|
|
if (empty($this->conversation)) {
|
|
|
|
// fallback to the atom:link rel="ostatus:conversation" element
|
2017-05-02 17:58:22 +01:00
|
|
|
$this->conversation = ActivityUtils::getLink($element, 'ostatus:'.self::CONVERSATION);
|
2014-11-27 12:39:38 +00:00
|
|
|
}
|
2010-03-20 21:06:22 +00:00
|
|
|
|
|
|
|
// Multiple attention links allowed
|
|
|
|
|
|
|
|
$links = $element->getElementsByTagNameNS(ActivityUtils::ATOM, ActivityUtils::LINK);
|
|
|
|
|
|
|
|
for ($i = 0; $i < $links->length; $i++) {
|
|
|
|
$link = $links->item($i);
|
|
|
|
|
2013-10-28 21:21:14 +00:00
|
|
|
$linkRel = $link->getAttribute(ActivityUtils::REL);
|
|
|
|
$linkHref = $link->getAttribute(self::HREF);
|
|
|
|
if ($linkRel == self::MENTIONED && $linkHref !== '') {
|
|
|
|
$this->attention[$linkHref] = $link->getAttribute(ActivityContext::OBJECTTYPE);
|
2010-03-20 21:06:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parse location given as a GeoRSS-simple point, if provided.
|
|
|
|
* http://www.georss.org/simple
|
|
|
|
*
|
|
|
|
* @param feed item $entry
|
|
|
|
* @return mixed Location or false
|
|
|
|
*/
|
|
|
|
function getLocation($dom)
|
|
|
|
{
|
|
|
|
$points = $dom->getElementsByTagNameNS(self::GEORSS, self::POINT);
|
|
|
|
|
|
|
|
for ($i = 0; $i < $points->length; $i++) {
|
|
|
|
$point = $points->item($i)->textContent;
|
|
|
|
return self::locationFromPoint($point);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// XXX: Move to ActivityUtils or Location?
|
|
|
|
static function locationFromPoint($point)
|
|
|
|
{
|
|
|
|
$point = str_replace(',', ' ', $point); // per spec "treat commas as whitespace"
|
|
|
|
$point = preg_replace('/\s+/', ' ', $point);
|
|
|
|
$point = trim($point);
|
|
|
|
$coords = explode(' ', $point);
|
|
|
|
if (count($coords) == 2) {
|
|
|
|
list($lat, $lon) = $coords;
|
|
|
|
if (is_numeric($lat) && is_numeric($lon)) {
|
|
|
|
common_log(LOG_INFO, "Looking up location for $lat $lon from georss point");
|
|
|
|
return Location::fromLatLon($lat, $lon);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
common_log(LOG_ERR, "Ignoring bogus georss:point value $point");
|
|
|
|
return null;
|
|
|
|
}
|
2011-02-18 08:51:08 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns context (StatusNet stuff) as an array suitable for serializing
|
|
|
|
* in JSON. Right now context stuff is an extension to Activity.
|
|
|
|
*
|
|
|
|
* @return array the context
|
|
|
|
*/
|
2011-02-18 18:02:41 +00:00
|
|
|
|
2011-02-18 08:51:08 +00:00
|
|
|
function asArray()
|
|
|
|
{
|
|
|
|
$context = array();
|
|
|
|
|
2011-02-19 02:52:49 +00:00
|
|
|
$context['inReplyTo'] = $this->getInReplyToArray();
|
2011-02-18 08:51:08 +00:00
|
|
|
$context['conversation'] = $this->conversation;
|
2017-05-02 17:58:22 +01:00
|
|
|
$context['conversation_url'] = $this->conversation_url;
|
2011-02-18 08:51:08 +00:00
|
|
|
|
|
|
|
return array_filter($context);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns an array of arrays representing Activity Objects (intended to be
|
|
|
|
* serialized in JSON) that represent WHO the Activity is supposed to
|
|
|
|
* be received by. This is not really specified but appears in an example
|
|
|
|
* of the current spec as an extension. We might want to figure out a JSON
|
|
|
|
* serialization for OStatus and use that to express mentions instead.
|
|
|
|
*
|
|
|
|
* XXX: People's ideas on how to do this are all over the place
|
|
|
|
*
|
|
|
|
* @return array the array of recipients
|
|
|
|
*/
|
2011-02-18 18:02:41 +00:00
|
|
|
|
2011-02-18 08:51:08 +00:00
|
|
|
function getToArray()
|
|
|
|
{
|
|
|
|
$tos = array();
|
|
|
|
|
2013-10-28 21:21:14 +00:00
|
|
|
foreach ($this->attention as $attnUrl => $attnType) {
|
2011-02-18 18:02:41 +00:00
|
|
|
$to = array(
|
2013-10-28 21:21:14 +00:00
|
|
|
'objectType' => $attnType, // can be empty
|
|
|
|
'id' => $attnUrl,
|
2011-02-18 18:02:41 +00:00
|
|
|
);
|
2011-02-18 08:51:08 +00:00
|
|
|
$tos[] = $to;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $tos;
|
|
|
|
}
|
|
|
|
|
2011-02-19 02:52:49 +00:00
|
|
|
/**
|
|
|
|
* Return an array for the notices this notice is a reply to
|
|
|
|
* suitable for serializing as JSON note objects.
|
|
|
|
*
|
|
|
|
* @return array the array of notes
|
2011-02-18 18:02:41 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
function getInReplyToArray()
|
|
|
|
{
|
2011-02-19 02:52:49 +00:00
|
|
|
if (empty($this->replyToID) && empty($this->replyToUrl)) {
|
|
|
|
return null;
|
|
|
|
}
|
2011-02-18 18:02:41 +00:00
|
|
|
|
2011-02-19 02:52:49 +00:00
|
|
|
$replyToObj = array('objectType' => 'note');
|
2011-02-18 18:02:41 +00:00
|
|
|
|
2011-02-19 02:52:49 +00:00
|
|
|
// XXX: Possibly shorten this to just the numeric ID?
|
|
|
|
// Currently, it's the full URI of the notice.
|
|
|
|
if (!empty($this->replyToID)) {
|
|
|
|
$replyToObj['id'] = $this->replyToID;
|
|
|
|
}
|
2011-02-18 18:02:41 +00:00
|
|
|
if (!empty($this->replyToUrl)) {
|
|
|
|
$replyToObj['url'] = $this->replyToUrl;
|
|
|
|
}
|
|
|
|
|
2011-02-19 02:52:49 +00:00
|
|
|
return $replyToObj;
|
2011-02-18 18:02:41 +00:00
|
|
|
}
|
|
|
|
|
2010-03-20 21:06:22 +00:00
|
|
|
}
|
2011-02-18 18:02:41 +00:00
|
|
|
|