streams/Zotlabs/Lib/Activity.php

2810 lines
74 KiB
PHP
Raw Normal View History

2018-05-30 04:08:52 +00:00
<?php
namespace Zotlabs\Lib;
use Zotlabs\Web\HTTPSig;
use Zotlabs\Access\Permissions;
use Zotlabs\Access\PermissionRoles;
2018-08-23 06:04:37 +00:00
use Zotlabs\Daemon\Master;
2018-05-30 04:08:52 +00:00
class Activity {
2018-09-19 23:25:05 +00:00
static $ACTOR_CACHE_DAYS = 3;
2018-05-30 04:08:52 +00:00
static function encode_object($x) {
if(($x) && (! is_array($x)) && (substr(trim($x),0,1)) === '{' ) {
2018-05-30 04:08:52 +00:00
$x = json_decode($x,true);
}
if($x['type'] === ACTIVITY_OBJ_PERSON) {
return self::fetch_person($x);
}
if($x['type'] === ACTIVITY_OBJ_PROFILE) {
return self::fetch_profile($x);
}
2018-07-12 01:02:25 +00:00
if(in_array($x['type'], [ ACTIVITY_OBJ_NOTE, ACTIVITY_OBJ_ARTICLE ] )) {
2018-05-30 04:08:52 +00:00
return self::fetch_item($x);
}
if($x['type'] === ACTIVITY_OBJ_THING) {
return self::fetch_thing($x);
}
return $x;
2018-05-30 04:08:52 +00:00
}
static function fetch($url,$channel = null) {
$redirects = 0;
if(! check_siteallowed($url)) {
logger('blacklisted: ' . $url);
return null;
}
if(! $channel) {
$channel = get_sys_channel();
}
if($channel) {
$m = parse_url($url);
$headers = [
'Accept' => 'application/activity+json, application/ld+json; profile="https://www.w3.org/ns/activitystreams"',
'Host' => $m['host'],
'(request-target)' => 'get ' . get_request_string($url),
'Date' => datetime_convert('UTC','UTC','now','D, d M Y H:i:s') . ' UTC'
];
$h = HTTPSig::create_sig($headers,$channel['channel_prvkey'],channel_url($channel),false);
}
else {
$h = [ 'Accept: application/activity+json, application/ld+json; profile="https://www.w3.org/ns/activitystreams"' ];
}
logger('fetch: ' . $url, LOGGER_DEBUG);
$x = z_fetch_url($url, true, $redirects, [ 'headers' => $h ] );
if($x['success']) {
$y = json_decode($x['body'],true);
logger('returned: ' . json_encode($y,JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES));
return json_decode($x['body'], true);
}
else {
logger('fetch failed: ' . $url);
}
return null;
}
2018-05-30 04:08:52 +00:00
static function fetch_person($x) {
return self::fetch_profile($x);
}
static function fetch_profile($x) {
$r = q("select * from xchan where xchan_url like '%s' limit 1",
dbesc($x['id'] . '/%')
);
if(! $r) {
$r = q("select * from xchan where xchan_hash = '%s' limit 1",
dbesc($x['id'])
);
}
if(! $r)
return [];
return self::encode_person($r[0],false);
2018-05-30 04:08:52 +00:00
}
static function fetch_thing($x) {
$r = q("select * from obj where obj_type = %d and obj_obj = '%s' limit 1",
intval(TERM_OBJ_THING),
dbesc($x['id'])
);
if(! $r)
return [];
$x = [
'type' => 'Object',
'id' => z_root() . '/thing/' . $r[0]['obj_obj'],
'name' => $r[0]['obj_term']
];
if($r[0]['obj_image'])
$x['image'] = $r[0]['obj_image'];
return $x;
}
static function fetch_item($x) {
if (array_key_exists('source',$x)) {
// This item is already processed and encoded
return $x;
}
2018-05-30 04:08:52 +00:00
$r = q("select * from item where mid = '%s' limit 1",
dbesc($x['id'])
);
if($r) {
xchan_query($r,true);
$r = fetch_post_tags($r,true);
2018-10-04 02:10:52 +00:00
return self::encode_item($r[0],((defined('NOMADIC')) ? false : true));
2018-05-30 04:08:52 +00:00
}
}
2018-08-20 03:39:23 +00:00
static function encode_item_collection($items,$id,$type,$activitypub = false) {
2018-05-30 04:08:52 +00:00
$ret = [
'id' => z_root() . '/' . $id,
'type' => $type,
'totalItems' => count($items),
];
if($items) {
$x = [];
foreach($items as $i) {
2018-09-10 03:55:51 +00:00
$m = get_iconfig($i['id'],'activitypub','rawmsg');
if($m) {
$t = json_decode($m,true);
}
else {
$t = self::encode_activity($i,$activitypub);
}
2018-05-30 04:08:52 +00:00
if($t)
$x[] = $t;
}
if($type === 'OrderedCollection')
$ret['orderedItems'] = $x;
else
$ret['items'] = $x;
}
return $ret;
}
static function encode_follow_collection($items,$id,$type,$extra = null) {
$ret = [
'id' => z_root() . '/' . $id,
'type' => $type,
'totalItems' => count($items),
];
if($extra)
$ret = array_merge($ret,$extra);
if($items) {
$x = [];
foreach($items as $i) {
if($i['xchan_url']) {
$x[] = $i['xchan_url'];
}
}
if($type === 'OrderedCollection')
$ret['orderedItems'] = $x;
else
$ret['items'] = $x;
}
return $ret;
}
2018-08-20 03:39:23 +00:00
static function encode_item($i, $activitypub = false) {
2018-05-30 04:08:52 +00:00
$ret = [];
$reply = false;
$is_directmessage = false;
2018-05-30 04:08:52 +00:00
$objtype = self::activity_obj_mapper($i['obj_type']);
2018-05-30 04:08:52 +00:00
if(intval($i['item_deleted'])) {
$ret['type'] = 'Tombstone';
$ret['formerType'] = $objtype;
2018-09-04 01:22:31 +00:00
$ret['id'] = $i['mid'];
$ret['to'] = [ ACTIVITY_PUBLIC_INBOX ];
2018-05-30 04:08:52 +00:00
return $ret;
}
$ret['type'] = $objtype;
2018-05-30 04:08:52 +00:00
2018-08-20 03:39:23 +00:00
/**
* If the destination is activitypub, see if the content needs conversion to
* Mastodon "quirks" mode. This will be the case if there is any markup beyond
* links or images OR if the number of images exceeds 1. This content may be
* purified into oblivion when using the Note type so turn it into an Article.
*/
$convert_to_article = false;
$images = false;
if($activitypub && $ret['type'] === 'Note') {
2018-10-04 09:19:08 +00:00
2018-08-20 03:39:23 +00:00
$bbtags = false;
$num_bbtags = preg_match_all('/\[\/([a-z]+)\]/ism',$i['body'],$bbtags,PREG_SET_ORDER);
2018-08-20 03:39:23 +00:00
if($num_bbtags) {
2018-08-20 03:39:23 +00:00
foreach($bbtags as $t) {
if((! $t[1]) || (in_array($t[1],['url','zrl','img','zmg']))) {
2018-08-20 03:39:23 +00:00
continue;
}
$convert_to_article = true;
}
}
2018-10-04 09:19:08 +00:00
2018-08-20 03:39:23 +00:00
$has_images = preg_match_all('/\[[zi]mg(.*?)\](.*?)\[/ism',$i['body'],$images,PREG_SET_ORDER);
2018-10-04 09:19:08 +00:00
2018-08-20 03:39:23 +00:00
if($has_images > 1) {
$convert_to_article = true;
}
if($convert_to_article) {
$ret['type'] = 'Article';
}
}
2018-09-04 01:22:31 +00:00
$ret['id'] = $i['mid'];
2018-05-30 04:08:52 +00:00
$ret['published'] = datetime_convert('UTC','UTC',$i['created'],ATOM_TIME);
2018-08-20 03:39:23 +00:00
if($i['created'] !== $i['edited']) {
2018-05-30 04:08:52 +00:00
$ret['updated'] = datetime_convert('UTC','UTC',$i['edited'],ATOM_TIME);
2018-08-20 03:39:23 +00:00
}
2018-05-30 04:08:52 +00:00
if($i['app']) {
$ret['instrument'] = [ 'type' => 'Service', 'name' => $i['app'] ];
}
if($i['location'] || $i['coord']) {
$ret['location'] = [ 'type' => 'Place' ];
if($i['location']) {
$ret['location']['name'] = $i['location'];
}
if($i['coord']) {
$l = explode(' ',$i['coord']);
$ret['location']['latitude'] = $l[0];
$ret['location']['longitude'] = $l[1];
}
}
2018-11-19 05:53:09 +00:00
$ret['inheritPrivacy'] = true;
2018-05-30 04:08:52 +00:00
$ret['attributedTo'] = $i['author']['xchan_url'];
2018-09-05 01:48:48 +00:00
2018-09-04 01:22:31 +00:00
if($i['mid'] !== $i['parent_mid']) {
2018-12-02 22:39:58 +00:00
$ret['inReplyTo'] = $i['thr_parent'];
$cnv = get_iconfig($i['parent'],'ostatus','conversation');
2018-12-02 22:39:58 +00:00
if(! $cnv) {
$cnv = $ret['parent_mid'];
}
$reply = true;
if($i['item_private']) {
$d = q("select xchan_url, xchan_addr, xchan_name from item left join xchan on xchan_hash = author_xchan where id = %d limit 1",
intval($i['parent'])
);
if($d) {
$recips = get_iconfig($i['parent'], 'activitypub', 'recips');
if(in_array($i['author']['xchan_url'], $recips['to'])) {
$reply_url = $d[0]['xchan_url'];
$is_directmessage = true;
}
else {
$reply_url = z_root() . '/followers/' . substr($i['author']['xchan_addr'],0,strpos($i['author']['xchan_addr'],'@'));
}
$reply_addr = (($d[0]['xchan_addr']) ? $d[0]['xchan_addr'] : $d[0]['xchan_name']);
}
}
}
if(! $cnv) {
$cnv = get_iconfig($i,'ostatus','conversation');
}
if($cnv) {
$ret['conversation'] = $cnv;
2018-05-30 04:08:52 +00:00
}
2018-07-09 05:28:15 +00:00
if($i['mimetype'] === 'text/bbcode') {
if($i['title'])
2018-08-20 03:39:23 +00:00
$ret['name'] = $i['title'];
if($i['summary'])
$ret['summary'] = bbcode($i['summary'], [ 'export' => true ]);
$ret['content'] = bbcode($i['body'], [ 'export' => true ]);
2018-08-20 03:39:23 +00:00
$ret['source'] = [ 'content' => $i['body'], 'summary' => $i['summary'], 'mediaType' => 'text/bbcode' ];
2018-05-30 04:08:52 +00:00
}
2018-06-26 03:55:53 +00:00
$actor = self::encode_person($i['author'],false);
2018-05-30 04:08:52 +00:00
if($actor)
$ret['actor'] = $actor;
else
return [];
$t = self::encode_taxonomy($i);
if($t) {
$ret['tag'] = $t;
}
$a = self::encode_attachment($i);
if($a) {
$ret['attachment'] = $a;
}
2018-08-20 03:39:23 +00:00
if($activitypub && $has_images && $ret['type'] === 'Note') {
$img = [];
foreach($images as $match) {
$img[] = [ 'type' => 'Image', 'url' => $match[2] ];
}
if(! $ret['attachment'])
$ret['attachment'] = [];
$ret['attachment'] = array_merge($img,$ret['attachment']);
}
if($activitypub) {
if($i['item_private']) {
if($reply) {
if($i['author_xchan'] === $i['owner_xchan']) {
$m = self::map_acl($i,(($i['allow_gid']) ? false : true));
$ret['tag'] = (($ret['tag']) ? array_merge($ret['tag'],$m) : $m);
}
else {
if($is_directmessage) {
$m = [
'type' => 'Mention',
'href' => $reply_url,
'name' => '@' . $reply_addr
];
$ret['tag'] = (($ret['tag']) ? array_merge($ret['tag'],$m) : $m);
}
else {
$ret['to'] = [ $reply_url ];
}
}
}
else {
/* Add mentions only if the targets are individuals */
$m = self::map_acl($i,(($i['allow_gid']) ? false : true));
$ret['tag'] = (($ret['tag']) ? array_merge($ret['tag'],$m) : $m);
}
}
else {
if($reply) {
$ret['to'] = [ z_root() . '/followers/' . substr($i['author']['xchan_addr'],0,strpos($i['author']['xchan_addr'],'@')) ];
$ret['cc'] = [ ACTIVITY_PUBLIC_INBOX ];
}
else {
$ret['to'] = [ ACTIVITY_PUBLIC_INBOX ];
$ret['cc'] = [ z_root() . '/followers/' . substr($i['author']['xchan_addr'],0,strpos($i['author']['xchan_addr'],'@')) ];
}
}
$mentions = self::map_mentions($i);
if(count($mentions) > 0) {
if(! $ret['to']) {
$ret['to'] = $mentions;
}
else {
$ret['to'] = array_merge($ret['to'], $mentions);
}
}
}
2018-05-30 04:08:52 +00:00
return $ret;
}
static function decode_taxonomy($item) {
$ret = [];
2018-10-19 00:37:34 +00:00
if($item['tag'] && is_array($item['tag'])) {
2018-05-30 04:08:52 +00:00
foreach($item['tag'] as $t) {
if(! array_key_exists('type',$t))
$t['type'] = 'Hashtag';
switch($t['type']) {
case 'Hashtag':
$ret[] = [ 'ttype' => TERM_HASHTAG, 'url' => $t['href'], 'term' => escape_tags((substr($t['name'],0,1) === '#') ? substr($t['name'],1) : $t['name']) ];
2018-05-30 04:08:52 +00:00
break;
case 'Mention':
$mention_type = substr($t['name'],0,1);
if($mention_type === '!') {
$ret[] = [ 'ttype' => TERM_FORUM, 'url' => $t['href'], 'term' => escape_tags(substr($t['name'],1)) ];
}
else {
$ret[] = [ 'ttype' => TERM_MENTION, 'url' => $t['href'], 'term' => escape_tags((substr($t['name'],0,1) === '@') ? substr($t['name'],1) : $t['name']) ];
}
2018-05-30 04:08:52 +00:00
break;
2018-09-06 23:40:13 +00:00
case 'Emoji':
2018-09-07 05:52:48 +00:00
$ret[] = [ 'ttype' => TERM_EMOJI, 'url' => $t['icon']['url'], 'term' => escape_tags($t['name']) ];
2018-09-06 23:40:13 +00:00
break;
2018-05-30 04:08:52 +00:00
default:
break;
}
}
}
return $ret;
}
static function encode_taxonomy($item) {
$ret = [];
if($item['term']) {
foreach($item['term'] as $t) {
switch($t['ttype']) {
case TERM_HASHTAG:
// An id is required so if we don't have a url in the taxonomy, ignore it and keep going.
if($t['url']) {
$ret[] = [ 'id' => $t['url'], 'name' => '#' . $t['term'] ];
}
break;
case TERM_FORUM:
2018-11-08 02:24:22 +00:00
$term = self::lookup_term_addr($t['url'],$t['term']);
$ret[] = [ 'type' => 'Mention', 'href' => $t['url'], 'name' => '!' . (($term) ? $term : $t['term']) ];
break;
2018-05-30 04:08:52 +00:00
case TERM_MENTION:
2018-11-08 02:24:22 +00:00
$term = self::lookup_term_addr($t['url'],$t['term']);
$ret[] = [ 'type' => 'Mention', 'href' => $t['url'], 'name' => '@' . (($term) ? $term : $t['term']) ];
2018-05-30 04:08:52 +00:00
break;
default:
break;
}
}
}
return $ret;
}
2018-11-08 02:24:22 +00:00
static function lookup_term_addr($url,$name) {
// The visible mention in our activities is always the full name.
// In the object taxonomy change this to the webfinger handle in case
// platforms expect the Mastodon form in order to generate notifications
// Try a couple of different things in case the url provided isn't the canonical id.
// If all else fails, try to match the name.
$r = false;
if($url) {
$r = q("select xchan_addr from xchan where ( xchan_url = '%s' OR xchan_hash = '%s' ) limit 1",
dbesc($url),
dbesc($url)
);
if($r) {
return $r[0]['xchan_addr'];
}
}
if($name) {
$r = q("select xchan_addr from xchan where xchan_name = '%s' limit 1",
dbesc($name)
);
if($r) {
return $r[0]['xchan_addr'];
}
}
return EMPTY_STR;
}
static function lookup_term_url($url) {
// The xchan_url for mastodon is a text/html rendering. This is called from map_mentions where we need
// to convert the mention url to an ActivityPub id. If this fails for any reason, return the url we have
2018-11-12 03:16:22 +00:00
$r = q("select hubloc_id_url from hubloc left join xchan on hubloc_hash = xchan_hash where xchan_url = '%s' and hubloc_primary = 1 limit 1",
2018-11-08 02:24:22 +00:00
dbesc($url)
);
if($r) {
return $r[0]['hubloc_id_url'];
}
return EMPTY_STR;
}
2018-05-30 04:08:52 +00:00
static function encode_attachment($item) {
$ret = [];
if($item['attach']) {
$atts = json_decode($item['attach'],true);
if($atts) {
foreach($atts as $att) {
if(strpos($att['type'],'image')) {
$ret[] = [ 'type' => 'Image', 'url' => $att['href'] ];
}
else {
$ret[] = [ 'type' => 'Link', 'mediaType' => $att['type'], 'href' => $att['href'] ];
}
}
}
}
return $ret;
}
static function decode_attachment($item) {
$ret = [];
if($item['attachment']) {
foreach($item['attachment'] as $att) {
$entry = [];
if($att['href'])
$entry['href'] = $att['href'];
elseif($att['url'])
$entry['href'] = $att['url'];
if($att['mediaType'])
$entry['type'] = $att['mediaType'];
elseif($att['type'] === 'Image')
$entry['type'] = 'image/jpeg';
if($entry)
$ret[] = $entry;
}
}
return $ret;
}
2018-08-20 03:39:23 +00:00
static function encode_activity($i,$activitypub = false) {
2018-05-30 04:08:52 +00:00
$ret = [];
$reply = false;
if(intval($i['item_deleted'])) {
$ret['type'] = 'Delete';
$ret['id'] = str_replace('/item/','/activity/',$i['mid']) . '#delete';
$actor = self::encode_person($i['author'],false);
if($actor)
$ret['actor'] = $actor;
else
return [];
if($i['obj']) {
if(! is_array($i['obj'])) {
$i['obj'] = json_decode($i['obj'],true);
}
$obj = self::encode_object($i['obj']);
if($obj)
$ret['object'] = $obj;
else
return [];
}
else {
$obj = self::encode_item($i,$activitypub);
if($obj)
$ret['object'] = $obj;
else
return [];
}
$ret['to'] = [ ACTIVITY_PUBLIC_INBOX ];
2018-05-30 04:08:52 +00:00
return $ret;
2018-05-30 04:08:52 +00:00
}
2018-05-30 06:30:18 +00:00
$ret['type'] = self::activity_mapper($i['verb']);
2018-09-03 06:12:53 +00:00
if(strpos($i['mid'],z_root() . '/item/') !== false) {
2018-09-04 01:22:31 +00:00
$ret['id'] = str_replace('/item/','/activity/',$i['mid']);
}
2018-10-16 20:56:46 +00:00
elseif(strpos($i['mid'],z_root() . '/event/') !== false) {
$ret['id'] = str_replace('/event/','/activity/',$i['mid']);
}
2018-09-04 01:22:31 +00:00
else {
$ret['id'] = $i['mid'];
2018-09-03 06:12:53 +00:00
}
2018-05-30 04:08:52 +00:00
2018-08-20 03:39:23 +00:00
if($i['title']) {
$ret['name'] = $i['title'];
}
2018-08-20 03:39:23 +00:00
if($i['summary']) {
$ret['summary'] = bbcode($i['summary'], [ 'export' => true ]);
2018-08-20 03:39:23 +00:00
}
2018-05-30 04:08:52 +00:00
2018-08-01 02:39:50 +00:00
if($ret['type'] === 'Announce') {
2018-09-03 03:16:33 +00:00
$tmp = $i['body'];
$ret['content'] = bbcode($tmp, [ 'export' => true ]);
2018-08-01 02:39:50 +00:00
$ret['source'] = [
'content' => $i['body'],
'mediaType' => 'text/bbcode'
];
}
2018-05-30 04:08:52 +00:00
$ret['published'] = datetime_convert('UTC','UTC',$i['created'],ATOM_TIME);
if($i['created'] !== $i['edited'])
$ret['updated'] = datetime_convert('UTC','UTC',$i['edited'],ATOM_TIME);
if($i['app']) {
$ret['instrument'] = [ 'type' => 'Service', 'name' => $i['app'] ];
}
if($i['location'] || $i['coord']) {
$ret['location'] = [ 'type' => 'Place' ];
if($i['location']) {
$ret['location']['name'] = $i['location'];
}
if($i['coord']) {
$l = explode(' ',$i['coord']);
$ret['location']['latitude'] = $l[0];
$ret['location']['longitude'] = $l[1];
}
}
2018-12-02 22:39:58 +00:00
if($i['mid'] != $i['parent_mid']) {
$ret['inReplyTo'] = $i['thr_parent'];
$cnv = get_iconfig($i['parent'],'ostatus','conversation');
if(! $cnv) {
$cnv = $ret['parent_mid'];
}
2018-05-30 04:08:52 +00:00
$reply = true;
if($i['item_private']) {
$d = q("select xchan_url, xchan_addr, xchan_name from item left join xchan on xchan_hash = author_xchan where id = %d limit 1",
intval($i['parent'])
);
if($d) {
$is_directmessage = false;
$recips = get_iconfig($i['parent'], 'activitypub', 'recips');
if(in_array($i['author']['xchan_url'], $recips['to'])) {
$reply_url = $d[0]['xchan_url'];
$is_directmessage = true;
}
else {
$reply_url = z_root() . '/followers/' . substr($i['author']['xchan_addr'],0,strpos($i['author']['xchan_addr'],'@'));
}
$reply_addr = (($d[0]['xchan_addr']) ? $d[0]['xchan_addr'] : $d[0]['xchan_name']);
}
}
}
2018-12-02 22:39:58 +00:00
if(! $cnv) {
$cnv = get_iconfig($i,'ostatus','conversation');
}
if($cnv) {
$ret['conversation'] = $cnv;
}
2018-11-19 05:53:09 +00:00
$ret['inheritPrivacy'] = true;
2018-06-26 03:55:53 +00:00
$actor = self::encode_person($i['author'],false);
2018-05-30 04:08:52 +00:00
if($actor)
$ret['actor'] = $actor;
else
return [];
if($i['obj']) {
2018-07-12 01:02:25 +00:00
if(! is_array($i['obj'])) {
$i['obj'] = json_decode($i['obj'],true);
}
2018-05-30 04:08:52 +00:00
$obj = self::encode_object($i['obj']);
if($obj)
$ret['object'] = $obj;
else
return [];
}
else {
2018-10-04 02:10:52 +00:00
$obj = self::encode_item($i,$activitypub);
2018-05-30 04:08:52 +00:00
if($obj)
$ret['object'] = $obj;
else
return [];
}
if($i['target']) {
2018-07-12 01:02:25 +00:00
if(! is_array($i['target'])) {
$i['target'] = json_decode($i['target'],true);
}
2018-05-30 04:08:52 +00:00
$tgt = self::encode_object($i['target']);
if($tgt)
$ret['target'] = $tgt;
else
return [];
}
2018-08-20 03:39:23 +00:00
if($activitypub) {
if($i['item_private']) {
if($reply) {
if($i['author_xchan'] === $i['owner_xchan']) {
$m = self::map_acl($i,(($i['allow_gid']) ? false : true));
$ret['tag'] = (($ret['tag']) ? array_merge($ret['tag'],$m) : $m);
}
else {
if($is_directmessage) {
$m = [
'type' => 'Mention',
'href' => $reply_url,
'name' => '@' . $reply_addr
];
$ret['tag'] = (($ret['tag']) ? array_merge($ret['tag'],$m) : $m);
}
else {
$ret['to'] = [ $reply_url ];
}
}
}
else {
/* Add mentions only if the targets are individuals */
$m = self::map_acl($i,(($i['allow_gid']) ? false : true));
$ret['tag'] = (($ret['tag']) ? array_merge($ret['tag'],$m) : $m);
}
}
else {
if($reply) {
$ret['to'] = [ z_root() . '/followers/' . substr($i['author']['xchan_addr'],0,strpos($i['author']['xchan_addr'],'@')) ];
$ret['cc'] = [ ACTIVITY_PUBLIC_INBOX ];
}
else {
$ret['to'] = [ ACTIVITY_PUBLIC_INBOX ];
$ret['cc'] = [ z_root() . '/followers/' . substr($i['author']['xchan_addr'],0,strpos($i['author']['xchan_addr'],'@')) ];
}
}
$mentions = self::map_mentions($i);
if(count($mentions) > 0) {
if(! $ret['to']) {
$ret['to'] = $mentions;
2018-08-20 03:39:23 +00:00
}
else {
$ret['to'] = array_merge($ret['to'], $mentions);
2018-08-20 03:39:23 +00:00
}
}
}
2018-05-30 04:08:52 +00:00
return $ret;
}
static function map_mentions($i) {
if(! $i['term']) {
return [];
}
$list = [];
foreach ($i['term'] as $t) {
if($t['ttype'] == TERM_MENTION) {
2018-11-08 02:24:22 +00:00
$url = self::lookup_term_url($t['url']);
$list[] = (($url) ? $url : $t['url']);
2018-05-30 04:08:52 +00:00
}
}
return $list;
}
static function map_acl($i,$mentions = false) {
$private = false;
$list = [];
$x = collect_recipients($i,$private);
2018-08-20 03:39:23 +00:00
2018-05-30 04:08:52 +00:00
if($x) {
stringify_array_elms($x);
if(! $x)
return;
$strict = (($mentions) ? true : get_config('activitypub','compliance'));
$sql_extra = (($strict) ? " and xchan_network = 'activitypub' " : '');
$details = q("select xchan_url, xchan_addr, xchan_name from xchan where xchan_hash in (" . implode(',',$x) . ") $sql_extra");
if($details) {
foreach($details as $d) {
if($mentions) {
$list[] = [ 'type' => 'Mention', 'href' => $d['xchan_url'], 'name' => '@' . (($d['xchan_addr']) ? $d['xchan_addr'] : $d['xchan_name']) ];
}
else {
$list[] = $d['xchan_url'];
}
}
}
}
return $list;
}
2018-08-20 03:39:23 +00:00
static function encode_person($p, $extended = true, $activitypub = false) {
2018-06-26 03:55:53 +00:00
2018-10-05 01:40:57 +00:00
$ret = [];
2018-05-30 04:08:52 +00:00
if(! $p['xchan_url'])
2018-10-05 01:40:57 +00:00
return $ret;
2018-05-30 04:08:52 +00:00
2018-06-26 03:55:53 +00:00
if(! $extended) {
return $p['xchan_url'];
}
2018-10-05 01:40:57 +00:00
$c = ((array_key_exists('channel_id',$p)) ? $p : channelx_by_hash($p['xchan_hash']));
2018-05-30 04:08:52 +00:00
$ret['type'] = 'Person';
2018-10-05 01:40:57 +00:00
if($c) {
2018-12-12 23:39:17 +00:00
$role = PConfig::Get($c['channel_id'],'system','permissions_role');
2018-10-05 01:40:57 +00:00
if(strpos($role,'forum') !== false) {
$ret['type'] = 'Group';
}
}
2018-11-09 01:02:04 +00:00
$ret['id'] = ((strpos($p['xchan_hash'],'http') === 0) ? $p['xchan_hash'] : $p['xchan_url']);
2018-05-30 04:08:52 +00:00
if($p['xchan_addr'] && strpos($p['xchan_addr'],'@'))
$ret['preferredUsername'] = substr($p['xchan_addr'],0,strpos($p['xchan_addr'],'@'));
$ret['name'] = $p['xchan_name'];
$ret['updated'] = datetime_convert('UTC','UTC',$p['xchan_name_date'],ATOM_TIME);
2018-05-30 04:08:52 +00:00
$ret['icon'] = [
'type' => 'Image',
'mediaType' => (($p['xchan_photo_mimetype']) ? $p['xchan_photo_mimetype'] : 'image/png' ),
'updated' => datetime_convert('UTC','UTC',$p['xchan_photo_date'],ATOM_TIME),
2018-05-30 04:08:52 +00:00
'url' => $p['xchan_photo_l'],
'height' => 300,
'width' => 300,
2018-06-25 01:54:29 +00:00
];
2018-05-30 04:08:52 +00:00
$ret['url'] = [
2018-06-25 01:54:29 +00:00
[
'type' => 'Link',
'mediaType' => 'text/html',
'href' => $p['xchan_url']
],
[
'type' => 'Link',
'mediaType' => 'text/x-zot+json',
'href' => $p['xchan_url']
]
2018-05-30 04:08:52 +00:00
];
2018-08-29 09:19:32 +00:00
2018-08-20 03:39:23 +00:00
if($activitypub) {
2018-08-29 09:19:32 +00:00
2018-08-20 03:39:23 +00:00
if($c) {
$ret['inbox'] = z_root() . '/inbox/' . $c['channel_address'];
$ret['outbox'] = z_root() . '/outbox/' . $c['channel_address'];
$ret['followers'] = z_root() . '/followers/' . $c['channel_address'];
$ret['following'] = z_root() . '/following/' . $c['channel_address'];
$ret['endpoints'] = [ 'sharedInbox' => z_root() . '/inbox' ];
$ret['publicKey'] = [
'id' => $p['xchan_url'] . '/public_key_pem',
'owner' => $p['xchan_url'],
'publicKeyPem' => $p['xchan_pubkey']
];
$cp = get_cover_photo($c['channel_id'],'array');
if($cp) {
$ret['image'] = [
'type' => 'Image',
'mediaType' => $cp['type'],
'url' => $cp['url']
];
}
$dp = q("select about from profile where uid = %d and is_default = 1",
intval($c['channel_id'])
);
if($dp && $dp[0]['about']) {
$ret['summary'] = bbcode($dp[0]['about'],['export' => true ]);
}
2018-08-20 03:39:23 +00:00
}
else {
2018-11-22 03:04:42 +00:00
$collections = get_xconfig($p['xchan_hash'],'activitypub','collections',[]);
2018-08-20 03:39:23 +00:00
if($collections) {
$ret = array_merge($ret,$collections);
}
else {
$ret['inbox'] = null;
$ret['outbox'] = null;
}
}
}
2018-08-29 09:19:32 +00:00
else {
$ret['inbox'] = z_root() . '/nullbox';
$ret['outbox'] = z_root() . '/nullbox';
$ret['publicKey'] = [
'id' => $p['xchan_url'] . '/public_key_pem',
'owner' => $p['xchan_url'],
'publicKeyPem' => $p['xchan_pubkey']
];
}
2018-08-20 03:39:23 +00:00
$arr = [ 'xchan' => $p, 'encoded' => $ret, 'activitypub' => $activitypub ];
call_hooks('encode_person', $arr);
$ret = $arr['encoded'];
2018-05-30 04:08:52 +00:00
return $ret;
}
static function activity_mapper($verb) {
if(strpos($verb,'/') === false) {
return $verb;
}
$acts = [
'http://activitystrea.ms/schema/1.0/post' => 'Create',
'http://activitystrea.ms/schema/1.0/share' => 'Announce',
'http://activitystrea.ms/schema/1.0/update' => 'Update',
'http://activitystrea.ms/schema/1.0/like' => 'Like',
'http://activitystrea.ms/schema/1.0/favorite' => 'Like',
'http://purl.org/zot/activity/dislike' => 'Dislike',
'http://activitystrea.ms/schema/1.0/tag' => 'Add',
'http://activitystrea.ms/schema/1.0/follow' => 'Follow',
'http://activitystrea.ms/schema/1.0/unfollow' => 'Unfollow',
];
if(array_key_exists($verb,$acts) && $acts[$verb]) {
return $acts[$verb];
}
// Reactions will just map to normal activities
if(strpos($verb,ACTIVITY_REACT) !== false)
return 'Create';
if(strpos($verb,ACTIVITY_MOOD) !== false)
return 'Create';
if(strpos($verb,ACTIVITY_POKE) !== false)
return 'Activity';
// We should return false, however this will trigger an uncaught execption and crash
// the delivery system if encountered by the JSON-LDSignature library
logger('Unmapped activity: ' . $verb);
return 'Create';
// return false;
}
static function activity_obj_mapper($obj) {
if(strpos($obj,'/') === false) {
return $obj;
}
$objs = [
'http://activitystrea.ms/schema/1.0/note' => 'Note',
'http://activitystrea.ms/schema/1.0/comment' => 'Note',
'http://activitystrea.ms/schema/1.0/person' => 'Person',
'http://purl.org/zot/activity/profile' => 'Profile',
'http://activitystrea.ms/schema/1.0/photo' => 'Image',
'http://activitystrea.ms/schema/1.0/profile-photo' => 'Icon',
'http://activitystrea.ms/schema/1.0/event' => 'Event',
'http://activitystrea.ms/schema/1.0/wiki' => 'Document',
'http://purl.org/zot/activity/location' => 'Place',
'http://purl.org/zot/activity/chessgame' => 'Game',
'http://purl.org/zot/activity/tagterm' => 'zot:Tag',
'http://purl.org/zot/activity/thing' => 'Object',
'http://purl.org/zot/activity/file' => 'zot:File',
'http://purl.org/zot/activity/mood' => 'zot:Mood',
];
if(array_key_exists($obj,$objs)) {
return $objs[$obj];
}
logger('Unmapped activity object: ' . $obj);
return 'Note';
// return false;
}
static function follow($channel,$act) {
$contact = null;
$their_follow_id = null;
/*
*
* if $act->type === 'Follow', actor is now following $channel
* if $act->type === 'Accept', actor has approved a follow request from $channel
*
*/
$person_obj = $act->actor;
if($act->type === 'Follow') {
$their_follow_id = $act->id;
}
elseif($act->type === 'Accept') {
$my_follow_id = z_root() . '/follow/' . $contact['id'];
}
if(is_array($person_obj)) {
// store their xchan and hubloc
self::actor_store($person_obj['id'],$person_obj);
// Find any existing abook record
$r = q("select * from abook left join xchan on abook_xchan = xchan_hash where abook_xchan = '%s' and abook_channel = %d limit 1",
dbesc($person_obj['id']),
intval($channel['channel_id'])
);
if($r) {
$contact = $r[0];
}
}
$x = PermissionRoles::role_perms('social');
$p = Permissions::FilledPerms($x['perms_connect']);
// add tag_deliver permissions to remote groups
if(is_array($person_obj) && $person_obj['type'] === 'Group') {
$p['tag_deliver'] = 1;
}
$their_perms = Permissions::serialise($p);
2018-05-30 04:08:52 +00:00
2018-05-30 04:08:52 +00:00
if($contact && $contact['abook_id']) {
// A relationship of some form already exists on this site.
switch($act->type) {
case 'Follow':
// A second Follow request, but we haven't approved the first one
if($contact['abook_pending']) {
return;
}
// We've already approved them or followed them first
// Send an Accept back to them
2018-09-20 06:53:26 +00:00
set_abconfig($channel['channel_id'],$person_obj['id'],'activitypub','their_follow_id', $their_follow_id);
Master::Summon([ 'Notifier', 'permissions_accept', $contact['abook_id'] ]);
2018-05-30 04:08:52 +00:00
return;
case 'Accept':
// They accepted our Follow request - set default permissions
set_abconfig($channel['channel_id'],$contact['abook_xchan'],'system','their_perms',$their_perms);
2018-05-30 04:08:52 +00:00
$abook_instance = $contact['abook_instance'];
if(strpos($abook_instance,z_root()) === false) {
if($abook_instance)
$abook_instance .= ',';
$abook_instance .= z_root();
$r = q("update abook set abook_instance = '%s', abook_not_here = 0
where abook_id = %d and abook_channel = %d",
dbesc($abook_instance),
intval($contact['abook_id']),
intval($channel['channel_id'])
);
}
return;
default:
return;
}
}
// No previous relationship exists.
if($act->type === 'Accept') {
// This should not happen unless we deleted the connection before it was accepted.
return;
}
// From here on out we assume a Follow activity to somebody we have no existing relationship with
2018-09-20 06:53:26 +00:00
set_abconfig($channel['channel_id'],$person_obj['id'],'activitypub','their_follow_id', $their_follow_id);
2018-05-30 04:08:52 +00:00
// The xchan should have been created by actor_store() above
$r = q("select * from xchan where xchan_hash = '%s' and xchan_network = 'activitypub' limit 1",
dbesc($person_obj['id'])
);
if(! $r) {
logger('xchan not found for ' . $person_obj['id']);
return;
}
$ret = $r[0];
$p = Permissions::connect_perms($channel['channel_id']);
$my_perms = Permissions::serialise($p['perms']);
2018-05-30 04:08:52 +00:00
$automatic = $p['automatic'];
2018-12-12 23:39:17 +00:00
$closeness = PConfig::Get($channel['channel_id'],'system','new_abook_closeness',80);
2018-05-30 04:08:52 +00:00
$r = abook_store_lowlevel(
[
'abook_account' => intval($channel['channel_account_id']),
'abook_channel' => intval($channel['channel_id']),
'abook_xchan' => $ret['xchan_hash'],
'abook_closeness' => intval($closeness),
'abook_created' => datetime_convert(),
'abook_updated' => datetime_convert(),
'abook_connected' => datetime_convert(),
'abook_dob' => NULL_DATE,
'abook_pending' => intval(($automatic) ? 0 : 1),
'abook_instance' => z_root()
]
);
if($my_perms)
AbConfig::Set($channel['channel_id'],$ret['xchan_hash'],'system','my_perms',$my_perms);
2018-05-30 04:08:52 +00:00
if($their_perms)
2018-12-12 23:39:17 +00:00
AbConfig::Set($channel['channel_id'],$ret['xchan_hash'],'system','their_perms',$their_perms);
2018-05-30 04:08:52 +00:00
if($r) {
logger("New ActivityPub follower for {$channel['channel_name']}");
$new_connection = q("select * from abook left join xchan on abook_xchan = xchan_hash left join hubloc on hubloc_hash = xchan_hash where abook_channel = %d and abook_xchan = '%s' order by abook_created desc limit 1",
intval($channel['channel_id']),
dbesc($ret['xchan_hash'])
);
if($new_connection) {
Enotify::submit(
2018-05-30 04:08:52 +00:00
[
'type' => NOTIFY_INTRO,
'from_xchan' => $ret['xchan_hash'],
'to_xchan' => $channel['channel_hash'],
'link' => z_root() . '/connedit/' . $new_connection[0]['abook_id'],
]
);
if($my_perms && $automatic) {
// send an Accept for this Follow activity
Master::Summon([ 'Notifier', 'permissions_accept', $new_connection[0]['abook_id'] ]);
2018-05-30 04:08:52 +00:00
// Send back a Follow notification to them
Master::Summon([ 'Notifier', 'permissions_create', $new_connection[0]['abook_id'] ]);
2018-05-30 04:08:52 +00:00
}
$clone = array();
foreach($new_connection[0] as $k => $v) {
if(strpos($k,'abook_') === 0) {
$clone[$k] = $v;
}
}
unset($clone['abook_id']);
unset($clone['abook_account']);
unset($clone['abook_channel']);
$abconfig = load_abconfig($channel['channel_id'],$clone['abook_xchan']);
if($abconfig)
$clone['abconfig'] = $abconfig;
2018-06-05 01:40:11 +00:00
Libsync::build_sync_packet($channel['channel_id'], [ 'abook' => array($clone) ] );
2018-05-30 04:08:52 +00:00
}
}
/* If there is a default group for this channel and permissions are automatic, add this member to it */
if($channel['channel_default_group'] && $automatic) {
2018-07-03 23:40:54 +00:00
$g = Group::rec_byhash($channel['channel_id'],$channel['channel_default_group']);
2018-05-30 04:08:52 +00:00
if($g)
2018-07-03 23:40:54 +00:00
Group::member_add($channel['channel_id'],'',$ret['xchan_hash'],$g['id']);
2018-05-30 04:08:52 +00:00
}
return;
}
static function unfollow($channel,$act) {
$contact = null;
/* @FIXME This really needs to be a signed request. */
/* actor is unfollowing $channel */
$person_obj = $act->actor;
if(is_array($person_obj)) {
$r = q("select * from abook left join xchan on abook_xchan = xchan_hash where abook_xchan = '%s' and abook_channel = %d limit 1",
dbesc($person_obj['id']),
intval($channel['channel_id'])
);
if($r) {
// remove all permissions they provided
del_abconfig($channel['channel_id'],$r[0]['xchan_hash'],'system','their_perms',EMPTY_STR);
2018-05-30 04:08:52 +00:00
}
}
return;
}
static function actor_store($url,$person_obj) {
if(! is_array($person_obj))
return;
2018-10-09 04:27:35 +00:00
// logger('person_obj: ' . print_r($person_obj,true));
2018-09-19 23:25:05 +00:00
// We may have been passed a cached entry. If it is, and the cache duration has expired
// fetch a fresh copy before continuing.
2018-11-08 23:00:33 +00:00
if(array_key_exists('cached',$person_obj)) {
2018-11-09 01:02:04 +00:00
if(array_key_exists('updated',$person_obj) && datetime_convert('UTC','UTC',$person_obj['updated']) < datetime_convert('UTC','UTC','now - ' . self::$ACTOR_CACHE_DAYS . ' days')) {
$person_obj = self::fetch($url);
2018-11-08 23:00:33 +00:00
}
else {
return;
}
2018-09-19 23:25:05 +00:00
}
2018-11-09 01:02:04 +00:00
$url = $person_obj['id'];
2018-09-19 23:25:05 +00:00
if(! $url) {
return;
}
2018-05-30 04:08:52 +00:00
$name = $person_obj['name'];
if(! $name)
$name = $person_obj['preferredUsername'];
if(! $name)
$name = t('Unknown');
2018-09-17 04:41:24 +00:00
$username = $person_obj['preferredUsername'];
$h = parse_url($url);
if($h && $h['host']) {
$username .= '@' . $h['host'];
}
2018-05-30 04:08:52 +00:00
if($person_obj['icon']) {
if(is_array($person_obj['icon'])) {
if(array_key_exists('url',$person_obj['icon']))
$icon = $person_obj['icon']['url'];
else
$icon = $person_obj['icon'][0]['url'];
}
else
$icon = $person_obj['icon'];
}
if(! $icon) {
$icon = z_root() . '/' . get_default_profile_photo();
}
2018-05-30 04:08:52 +00:00
2018-09-24 00:03:10 +00:00
$links = false;
$profile = false;
if(is_array($person_obj['url'])) {
if(! array_key_exists(0,$person_obj['url'])) {
$links = [ $person_obj['url'] ];
}
else {
$links = $person_obj['url'];
}
}
if($links) {
foreach($links as $link) {
if(array_key_exists('mediaType',$link) && $link['mediaType'] === 'text/html') {
$profile = $link['href'];
}
}
if(! $profile) {
$profile = $links[0]['href'];
}
}
elseif(isset($person_obj['url']) && is_string($person_obj['url'])) {
2018-09-24 00:03:10 +00:00
$profile = $person_obj['url'];
}
if(! $profile) {
$profile = $url;
}
2018-05-30 04:08:52 +00:00
$inbox = $person_obj['inbox'];
2018-11-08 23:00:33 +00:00
// either an invalid identity or a cached entry of some kind which didn't get caught above
if((! $inbox) || strpos($inbox,z_root()) !== false) {
return;
}
2018-05-30 04:08:52 +00:00
$collections = [];
if($inbox) {
$collections['inbox'] = $inbox;
if($person_obj['outbox'])
$collections['outbox'] = $person_obj['outbox'];
if($person_obj['followers'])
$collections['followers'] = $person_obj['followers'];
if($person_obj['following'])
$collections['following'] = $person_obj['following'];
2018-09-18 06:34:33 +00:00
if($person_obj['endpoints'] && is_array($person_obj['endpoints']) && $person_obj['endpoints']['sharedInbox'])
2018-05-30 04:08:52 +00:00
$collections['sharedInbox'] = $person_obj['endpoints']['sharedInbox'];
}
2018-11-03 14:26:20 +00:00
if(isset($person_obj['publicKey']['publicKeyPem'])) {
//if(array_key_exists('publicKey',$person_obj) && array_key_exists('publicKeyPem',$person_obj['publicKey'])) {
2018-05-30 04:08:52 +00:00
if($person_obj['id'] === $person_obj['publicKey']['owner']) {
$pubkey = $person_obj['publicKey']['publicKeyPem'];
if(strstr($pubkey,'RSA ')) {
$pubkey = Keyutils::rsatopem($pubkey);
2018-05-30 04:08:52 +00:00
}
}
}
$r = q("select * from xchan where xchan_hash = '%s' limit 1",
dbesc($url)
);
if(! $r) {
// create a new record
$r = xchan_store_lowlevel(
[
'xchan_hash' => $url,
'xchan_guid' => $url,
'xchan_pubkey' => $pubkey,
2018-09-17 04:41:24 +00:00
'xchan_addr' => ((strpos($username,'@')) ? $username : ''),
'xchan_url' => $profile,
'xchan_name' => $name,
'xchan_name_date' => datetime_convert(),
'xchan_network' => 'activitypub',
'xchan_photo_date' => datetime_convert('UTC','UTC','1968-01-01'),
2018-09-06 23:59:19 +00:00
'xchan_photo_l' => z_root() . '/' . get_default_profile_photo(),
'xchan_photo_m' => z_root() . '/' . get_default_profile_photo(80),
'xchan_photo_s' => z_root() . '/' . get_default_profile_photo(48),
'xchan_photo_mimetype' => 'image/png',
2018-05-30 04:08:52 +00:00
]
);
}
else {
2018-09-19 23:25:05 +00:00
// Record exists. Cache existing records for a set number of days
2018-05-30 04:08:52 +00:00
// then refetch to catch updated profile photos, names, etc.
2018-09-19 23:25:05 +00:00
if($r[0]['xchan_name_date'] >= datetime_convert('UTC','UTC','now - ' . self::$ACTOR_CACHE_DAYS . ' days')) {
2018-05-30 04:08:52 +00:00
return;
2018-09-19 23:25:05 +00:00
}
2018-05-30 04:08:52 +00:00
// update existing record
2018-11-19 05:53:09 +00:00
$u = q("update xchan set xchan_name = '%s', xchan_pubkey = '%s', xchan_network = '%s', xchan_name_date = '%s' where xchan_hash = '%s'",
2018-05-30 04:08:52 +00:00
dbesc($name),
dbesc($pubkey),
dbesc('activitypub'),
dbesc(datetime_convert()),
dbesc($url)
);
if(strpos($username,'@') && ($r[0]['xchan_addr'] !== $username)) {
$r = q("update xchan set xchan_addr = '%s' where xchan_hash = '%s'",
dbesc($username),
dbesc($url)
);
}
2018-05-30 04:08:52 +00:00
}
if($collections) {
set_xconfig($url,'activitypub','collections',$collections);
}
2018-12-05 03:57:28 +00:00
$h = q("select * from hubloc where hubloc_hash = '%s' limit 1",
2018-05-30 04:08:52 +00:00
dbesc($url)
);
$m = parse_url($url);
if($m) {
$hostname = $m['host'];
$baseurl = $m['scheme'] . '://' . $m['host'] . (($m['port']) ? ':' . $m['port'] : '');
}
2018-12-05 03:57:28 +00:00
if(! $h) {
2018-05-30 04:08:52 +00:00
$r = hubloc_store_lowlevel(
[
'hubloc_guid' => $url,
'hubloc_hash' => $url,
2018-11-08 02:24:22 +00:00
'hubloc_id_url' => $url,
2018-09-17 04:41:24 +00:00
'hubloc_addr' => ((strpos($username,'@')) ? $username : ''),
2018-05-30 04:08:52 +00:00
'hubloc_network' => 'activitypub',
'hubloc_url' => $baseurl,
'hubloc_host' => $hostname,
'hubloc_callback' => $inbox,
'hubloc_updated' => datetime_convert(),
'hubloc_primary' => 1
]
);
}
else {
2018-12-05 03:57:28 +00:00
if(strpos($username,'@') && ($h[0]['hubloc_addr'] !== $username)) {
$r = q("update hubloc set hubloc_addr = '%s' where hubloc_hash = '%s'",
dbesc($username),
dbesc($url)
);
}
2018-12-05 03:57:28 +00:00
if($inbox !== $h[0]['hubloc_callback']) {
2018-12-04 11:24:41 +00:00
$r = q("update hubloc set hubloc_callback = '%s' where hubloc_hash = '%s'",
dbesc($inbox),
dbesc($url)
);
}
2018-11-21 05:50:41 +00:00
$r = q("update hubloc set hubloc_updated = '%s' where hubloc_hash = '%s'",
dbesc(datetime_convert()),
dbesc($url)
);
}
2018-05-30 04:08:52 +00:00
if(! $icon)
$icon = z_root() . '/' . get_default_profile_photo(300);
Master::Summon( [ 'Xchan_photo', bin2hex($icon), bin2hex($url) ] );
2018-05-30 04:08:52 +00:00
}
static function drop($channel,$observer,$act) {
$r = q("select * from item where mid = '%s' and uid = %d limit 1",
$act->obj['id'],
$channel['channel_id']
);
if(! $r) {
return;
}
if(in_array($observer,[ $r[0]['author_xchan'], $r[0]['owner_xchan'] ])) {
drop_item($r[0]['id'],false);
}
elseif(in_array($act->actor['id'],[ $r[0]['author_xchan'], $r[0]['owner_xchan'] ])) {
drop_item($r[0]['id'],false);
}
}
2018-05-30 04:08:52 +00:00
static function create_action($channel,$observer_hash,$act) {
2018-09-20 05:44:22 +00:00
if(in_array($act->obj['type'], [ 'Note', 'Article', 'Video', 'Audio', 'Image' ])) {
2018-05-30 04:08:52 +00:00
self::create_note($channel,$observer_hash,$act);
}
}
static function announce_action($channel,$observer_hash,$act) {
if(in_array($act->type, [ 'Announce' ])) {
self::announce_note($channel,$observer_hash,$act);
}
}
static function like_action($channel,$observer_hash,$act) {
2018-09-20 05:44:22 +00:00
if(in_array($act->obj['type'], [ 'Note', 'Article', 'Video', 'Audio', 'Image' ])) {
2018-05-30 04:08:52 +00:00
self::like_note($channel,$observer_hash,$act);
}
}
// sort function width decreasing
2018-09-17 05:04:11 +00:00
static function vid_sort($a,$b) {
2018-05-30 04:08:52 +00:00
if($a['width'] === $b['width'])
return 0;
return (($a['width'] > $b['width']) ? -1 : 1);
}
static function create_note($channel,$observer_hash,$act) {
$s = [];
// Mastodon only allows visibility in public timelines if the public inbox is listed in the 'to' field.
// They are hidden in the public timeline if the public inbox is listed in the 'cc' field.
// This is not part of the activitypub protocol - we might change this to show all public posts in pubstream at some point.
$pubstream = ((is_array($act->obj) && array_key_exists('to', $act->obj) && in_array(ACTIVITY_PUBLIC_INBOX, $act->obj['to'])) ? true : false);
$is_sys_channel = is_sys_channel($channel['channel_id']);
$parent = ((array_key_exists('inReplyTo',$act->obj)) ? urldecode($act->obj['inReplyTo']) : '');
if($parent) {
$r = q("select * from item where uid = %d and ( mid = '%s' or mid = '%s' ) limit 1",
intval($channel['channel_id']),
dbesc($parent),
dbesc(basename($parent))
);
if(! $r) {
logger('parent not found.');
return;
}
if($r[0]['owner_xchan'] === $channel['channel_hash']) {
if(! perm_is_allowed($channel['channel_id'],$observer_hash,'send_stream') && ! ($is_sys_channel && $pubstream)) {
logger('no comment permission.');
return;
}
}
$s['parent_mid'] = $r[0]['mid'];
$s['owner_xchan'] = $r[0]['owner_xchan'];
$s['author_xchan'] = $observer_hash;
}
else {
if(! perm_is_allowed($channel['channel_id'],$observer_hash,'send_stream') && ! ($is_sys_channel && $pubstream)) {
logger('no permission');
return;
}
$s['owner_xchan'] = $s['author_xchan'] = $observer_hash;
}
$abook = q("select * from abook where abook_xchan = '%s' and abook_channel = %d limit 1",
dbesc($observer_hash),
intval($channel['channel_id'])
);
2018-09-01 22:52:28 +00:00
if(is_array($act->obj)) {
$content = self::get_content($act->obj);
}
2018-05-30 04:08:52 +00:00
if(! $content) {
logger('no content');
return;
}
$s['aid'] = $channel['channel_account_id'];
$s['uid'] = $channel['channel_id'];
$s['mid'] = urldecode($act->obj['id']);
2018-06-04 00:49:48 +00:00
$s['plink'] = urldecode($act->obj['id']);
2018-05-30 04:08:52 +00:00
if($act->data['published']) {
$s['created'] = datetime_convert('UTC','UTC',$act->data['published']);
}
elseif($act->obj['published']) {
$s['created'] = datetime_convert('UTC','UTC',$act->obj['published']);
}
if($act->data['updated']) {
$s['edited'] = datetime_convert('UTC','UTC',$act->data['updated']);
}
elseif($act->obj['updated']) {
$s['edited'] = datetime_convert('UTC','UTC',$act->obj['updated']);
}
if(! $s['created'])
$s['created'] = datetime_convert();
if(! $s['edited'])
$s['edited'] = $s['created'];
if(! $s['parent_mid'])
$s['parent_mid'] = $s['mid'];
$s['title'] = self::bb_content($content,'name');
$s['summary'] = self::bb_content($content,'summary');
$s['body'] = self::bb_content($content,'content');
2018-05-30 04:08:52 +00:00
$s['verb'] = ACTIVITY_POST;
$s['obj_type'] = ACTIVITY_OBJ_NOTE;
2018-05-31 03:32:59 +00:00
$instrument = $act->get_property_obj('instrument');
if(! $instrument)
$instrument = $act->get_property_obj('instrument',$act->obj);
if($instrument && array_key_exists('type',$instrument)
&& $instrument['type'] === 'Service' && array_key_exists('name',$instrument)) {
$s['app'] = escape_tags($instrument['name']);
2018-05-31 03:32:59 +00:00
}
2018-05-30 04:08:52 +00:00
if($channel['channel_system']) {
if(! MessageFilter::evaluate($s,get_config('system','pubstream_incl'),get_config('system','pubstream_excl'))) {
2018-05-30 04:08:52 +00:00
logger('post is filtered');
return;
}
}
2018-12-11 23:57:15 +00:00
if(! post_is_importable($channel['channel_id'],$s,$abook[0])) {
logger('post is filtered');
return;
2018-05-30 04:08:52 +00:00
}
if($act->obj['conversation']) {
set_iconfig($s,'ostatus','conversation',$act->obj['conversation'],1);
}
$a = self::decode_taxonomy($act->obj);
if($a) {
$s['term'] = $a;
}
$a = self::decode_attachment($act->obj);
if($a) {
$s['attach'] = $a;
}
if($act->obj['type'] === 'Note' && $s['attach']) {
2018-10-11 04:36:55 +00:00
$s['body'] .= self::bb_attach($s['attach'],$s['body']);
2018-05-30 04:08:52 +00:00
}
// we will need a hook here to extract magnet links e.g. peertube
// right now just link to the largest mp4 we find that will fit in our
// standard content region
if($act->obj['type'] === 'Video') {
$vtypes = [
'video/mp4',
'video/ogg',
'video/webm'
];
$mps = [];
if(array_key_exists('url',$act->obj) && is_array($act->obj['url'])) {
foreach($act->obj['url'] as $vurl) {
if(in_array($vurl['mimeType'], $vtypes)) {
if(! array_key_exists('width',$vurl)) {
$vurl['width'] = 0;
}
$mps[] = $vurl;
}
}
}
if($mps) {
2018-09-17 05:04:11 +00:00
usort($mps,[ __CLASS__, 'vid_sort' ]);
2018-05-30 04:08:52 +00:00
foreach($mps as $m) {
if(intval($m['width']) < 500) {
$s['body'] .= "\n\n" . '[video]' . $m['href'] . '[/video]';
break;
}
}
}
}
if($act->recips && (! in_array(ACTIVITY_PUBLIC_INBOX,$act->recips)))
$s['item_private'] = 1;
set_iconfig($s,'activitypub','recips',$act->raw_recips);
if($parent) {
set_iconfig($s,'activitypub','rawmsg',$act->raw,1);
}
$x = null;
$r = q("select created, edited from item where mid = '%s' and uid = %d limit 1",
dbesc($s['mid']),
intval($s['uid'])
);
if($r) {
if($s['edited'] > $r[0]['edited']) {
$x = item_store_update($s);
}
else {
return;
}
}
else {
$x = item_store($s);
}
if(is_array($x) && $x['item_id']) {
if($parent) {
if($s['owner_xchan'] === $channel['channel_hash']) {
// We are the owner of this conversation, so send all received comments back downstream
Master::Summon(array('Notifier','comment-import',$x['item_id']));
2018-05-30 04:08:52 +00:00
}
$r = q("select * from item where id = %d limit 1",
intval($x['item_id'])
);
if($r) {
send_status_notifications($x['item_id'],$r[0]);
}
}
sync_an_item($channel['channel_id'],$x['item_id']);
}
}
2018-05-30 06:30:18 +00:00
2018-08-31 02:19:32 +00:00
static function share_bb($obj) {
// @fixme - error check and set defaults
$name = urlencode($obj['actor']['name']);
$profile = $obj['actor']['id'];
$photo = $obj['icon']['url'];
$s = "\r\n[share author='" . $name .
"' profile='" . $profile .
"' avatar='" . $photo .
"' link='" . $act->obj['id'] .
"' auth='" . ((is_matrix_url($act->obj['id'])) ? 'true' : 'false' ) .
"' posted='" . $act->obj['published'] .
"' message_id='" . $act->obj['id'] .
"']";
return $s;
}
2018-10-14 22:47:57 +00:00
static function get_actor_bbmention($id) {
$x = q("select * from hubloc left join xchan on hubloc_hash = xchan_hash where hubloc_hash = '%s' or hubloc_id_url = '%s' limit 1",
dbesc($id),
dbesc($id)
);
if($x) {
return sprintf('@[zrl=%s]%s[/zrl]',$x[0]['xchan_url'],$x[0]['xchan_name']);
}
return '@{' . $id . '}';
}
2018-08-31 02:19:32 +00:00
2018-05-30 06:30:18 +00:00
static function decode_note($act) {
2018-10-14 22:47:57 +00:00
$response_activity = false;
2018-05-30 06:30:18 +00:00
$s = [];
2018-09-01 22:52:28 +00:00
if(is_array($act->obj)) {
$content = self::get_content($act->obj);
}
2018-09-04 04:30:39 +00:00
$s['owner_xchan'] = $act->actor['id'];
2018-05-30 06:30:18 +00:00
$s['author_xchan'] = $act->actor['id'];
2018-09-03 03:16:33 +00:00
$s['mid'] = $act->obj['id'];
$s['parent_mid'] = $act->parent_id;
2018-05-30 06:30:18 +00:00
if($act->data['published']) {
$s['created'] = datetime_convert('UTC','UTC',$act->data['published']);
}
elseif($act->obj['published']) {
$s['created'] = datetime_convert('UTC','UTC',$act->obj['published']);
}
if($act->data['updated']) {
$s['edited'] = datetime_convert('UTC','UTC',$act->data['updated']);
}
elseif($act->obj['updated']) {
$s['edited'] = datetime_convert('UTC','UTC',$act->obj['updated']);
}
2018-10-16 20:56:46 +00:00
if(in_array($act->type, [ 'Like', 'Dislike', 'Flag', 'Block', 'Announce', 'Accept', 'Reject', 'TentativeAccept' ])) {
2018-10-14 22:47:57 +00:00
$response_activity = true;
2018-09-26 00:50:12 +00:00
$s['mid'] = $act->id;
$s['parent_mid'] = $act->obj['id'];
// over-ride the object timestamp with the activity
if($act->data['published']) {
$s['created'] = datetime_convert('UTC','UTC',$act->data['published']);
}
2018-10-14 22:47:57 +00:00
if($act->data['updated']) {
$s['edited'] = datetime_convert('UTC','UTC',$act->data['updated']);
}
2018-10-10 05:32:20 +00:00
$obj_actor = ((isset($act->obj['actor'])) ? $act->obj['actor'] : $act->get_actor('attributedTo', $act->obj));
2018-10-14 22:47:57 +00:00
// ensure we store the original actor
self::actor_store($obj_actor['id'],$obj_actor);
$mention = self::get_actor_bbmention($obj_actor['id']);
2018-10-10 05:32:20 +00:00
2018-09-26 00:50:12 +00:00
if($act->type === 'Like') {
2018-10-14 22:47:57 +00:00
$content['content'] = sprintf( t('Likes %1$s\'s %2$s'),$mention,$act->obj['type']) . "\n\n" . $content['content'];
2018-09-26 00:50:12 +00:00
}
if($act->type === 'Dislike') {
2018-10-14 22:47:57 +00:00
$content['content'] = sprintf( t('Doesn\'t like %1$s\'s %2$s'),$mention,$act->obj['type']) . "\n\n" . $content['content'];
2018-09-26 00:50:12 +00:00
}
2018-10-17 03:07:45 +00:00
if($act->type === 'Accept' && $act->obj['type'] === 'Event' ) {
$content['content'] = sprintf( t('Will attend %1$s\'s %2$s'),$mention,$act->obj['type']) . "\n\n" . $content['content'];
2018-10-16 20:56:46 +00:00
}
2018-10-17 03:07:45 +00:00
if($act->type === 'Reject' && $act->obj['type'] === 'Event' ) {
$content['content'] = sprintf( t('Will not attend %1$s\'s %2$s'),$mention,$act->obj['type']) . "\n\n" . $content['content'];
2018-10-16 20:56:46 +00:00
}
2018-10-17 03:07:45 +00:00
if($act->type === 'TentativeAccept' && $act->obj['type'] === 'Event' ) {
$content['content'] = sprintf( t('May attend %1$s\'s %2$s'),$mention,$act->obj['type']) . "\n\n" . $content['content'];
2018-10-16 20:56:46 +00:00
}
2018-10-14 22:47:57 +00:00
if($act->type === 'Announce') {
2018-10-15 02:02:15 +00:00
$content['content'] = sprintf( t('&#x1f501; Repeated %1$s\'s %2$s'), $mention, $act->obj['type']);
}
2018-08-01 02:50:39 +00:00
}
2018-09-03 03:16:33 +00:00
2018-09-26 00:50:12 +00:00
if(! $s['created'])
$s['created'] = datetime_convert();
if(! $s['edited'])
$s['edited'] = $s['created'];
2018-12-12 05:03:50 +00:00
$s['title'] = (($response_activity) ? EMPTY_STR : self::bb_content($content,'name'));
2018-09-03 03:16:33 +00:00
$s['summary'] = self::bb_content($content,'summary');
$s['body'] = ((self::bb_content($content,'bbcode') && (! $response_activity)) ? self::bb_content($content,'bbcode') : self::bb_content($content,'content'));
2018-08-01 02:50:39 +00:00
2019-02-01 21:11:20 +00:00
if($act->type === 'Tombstone' || ($act->type === 'Create' && $act->obj['type'] === 'Tombstone')) {
2018-08-04 02:26:16 +00:00
$s['item_deleted'] = 1;
}
2019-02-01 20:47:34 +00:00
$s['verb'] = self::activity_mapper($act->type);
$s['obj_type'] = self::activity_obj_mapper($act->obj['type']);
$s['obj'] = $act->obj;
2018-12-26 23:33:01 +00:00
if(is_array($obj) && array_path_exists('actor/id',$s['obj'])) {
$s['obj']['actor'] = $s['obj']['actor']['id'];
}
2019-02-01 20:47:34 +00:00
2018-12-26 23:33:01 +00:00
// @todo add target if present
2018-05-30 06:30:18 +00:00
$instrument = $act->get_property_obj('instrument');
2018-10-14 22:47:57 +00:00
if((! $instrument) && (! $response_activity)) {
$instrument = $act->get_property_obj('instrument',$act->obj);
2018-10-14 22:47:57 +00:00
}
if($instrument && array_key_exists('type',$instrument)
&& $instrument['type'] === 'Service' && array_key_exists('name',$instrument)) {
$s['app'] = escape_tags($instrument['name']);
}
2018-10-14 22:47:57 +00:00
if(! $response_activity) {
$a = self::decode_taxonomy($act->obj);
if($a) {
$s['term'] = $a;
foreach($a as $b) {
if($b['ttype'] === TERM_EMOJI) {
$s['title'] = str_replace($b['term'],'[img=16x16]' . $b['url'] . '[/img]',$s['title']);
$s['summary'] = str_replace($b['term'],'[img=16x16]' . $b['url'] . '[/img]',$s['summary']);
$s['body'] = str_replace($b['term'],'[img=16x16]' . $b['url'] . '[/img]',$s['body']);
}
2018-09-06 23:40:13 +00:00
}
}
2018-05-30 06:30:18 +00:00
2018-10-14 22:47:57 +00:00
$a = self::decode_attachment($act->obj);
if($a) {
$s['attach'] = $a;
}
2018-05-30 06:30:18 +00:00
}
2018-09-01 22:51:44 +00:00
if($act->obj['type'] === 'Note' && $s['attach']) {
2018-10-11 04:36:55 +00:00
$s['body'] .= self::bb_attach($s['attach'],$s['body']);
2018-09-01 22:51:44 +00:00
}
2018-05-30 06:30:18 +00:00
// we will need a hook here to extract magnet links e.g. peertube
// right now just link to the largest mp4 we find that will fit in our
// standard content region
2018-10-14 22:47:57 +00:00
if(! $response_activity) {
if($act->obj['type'] === 'Video') {
2018-05-30 06:30:18 +00:00
2018-10-14 22:47:57 +00:00
$vtypes = [
'video/mp4',
'video/ogg',
'video/webm'
];
2018-05-30 06:30:18 +00:00
2018-10-14 22:47:57 +00:00
$mps = [];
$ptr = null;
2018-09-26 06:50:35 +00:00
2018-10-14 22:47:57 +00:00
if(array_key_exists('url',$act->obj)) {
if(is_array($act->obj['url'])) {
if(array_key_exists(0,$act->obj['url'])) {
$ptr = $act->obj['url'];
}
else {
$ptr = [ $act->obj['url'] ];
}
foreach($ptr as $vurl) {
// peertube uses the non-standard element name 'mimeType' here
if(array_key_exists('mimeType',$vurl)) {
if(in_array($vurl['mimeType'], $vtypes)) {
if(! array_key_exists('width',$vurl)) {
$vurl['width'] = 0;
}
$mps[] = $vurl;
}
}
elseif(array_key_exists('mediaType',$vurl)) {
if(in_array($vurl['mediaType'], $vtypes)) {
if(! array_key_exists('width',$vurl)) {
$vurl['width'] = 0;
}
$mps[] = $vurl;
2018-10-14 22:47:57 +00:00
}
2018-09-26 06:50:35 +00:00
}
2018-05-30 06:30:18 +00:00
}
}
2018-10-14 22:47:57 +00:00
if($mps) {
usort($mps,[ __CLASS__, 'vid_sort' ]);
foreach($mps as $m) {
2018-10-30 22:42:52 +00:00
if(intval($m['width']) < 500 && self::media_not_in_body($m['href'],$s['body'])) {
2018-10-14 22:47:57 +00:00
$s['body'] .= "\n\n" . '[video]' . $m['href'] . '[/video]';
break;
}
2018-09-26 06:50:35 +00:00
}
2018-05-30 06:30:18 +00:00
}
2018-10-30 22:42:52 +00:00
elseif(is_string($act->obj['url']) && self::media_not_in_body($act->obj['url'],$s['body'])) {
2018-10-14 22:47:57 +00:00
$s['body'] .= "\n\n" . '[video]' . $act->obj['url'] . '[/video]';
}
2018-09-26 06:50:35 +00:00
}
2018-05-30 06:30:18 +00:00
}
2018-10-14 22:47:57 +00:00
if($act->obj['type'] === 'Audio') {
2018-09-20 05:44:22 +00:00
2018-10-14 22:47:57 +00:00
$atypes = [
'audio/mpeg',
'audio/ogg',
'audio/wav'
];
2018-09-20 05:44:22 +00:00
2018-10-14 22:47:57 +00:00
$ptr = null;
2018-09-26 06:50:35 +00:00
2018-10-14 22:47:57 +00:00
if(array_key_exists('url',$act->obj)) {
if(is_array($act->obj['url'])) {
if(array_key_exists(0,$act->obj['url'])) {
$ptr = $act->obj['url'];
}
else {
$ptr = [ $act->obj['url'] ];
}
foreach($ptr as $vurl) {
2018-10-30 22:42:52 +00:00
if(in_array($vurl['mediaType'], $atypes) && self::media_not_in_body($vurl['href'],$s['body'])) {
2018-10-14 22:47:57 +00:00
$s['body'] .= "\n\n" . '[audio]' . $vurl['href'] . '[/audio]';
break;
}
2018-09-26 06:50:35 +00:00
}
}
2018-10-30 22:42:52 +00:00
elseif(is_string($act->obj['url']) && self::media_not_in_body($act->obj['url'],$s['body'])) {
2018-10-14 22:47:57 +00:00
$s['body'] .= "\n\n" . '[audio]' . $act->obj['url'] . '[/audio]';
}
2018-09-26 06:50:35 +00:00
}
2018-10-14 22:47:57 +00:00
}
2018-09-20 05:44:22 +00:00
2019-01-25 03:05:43 +00:00
if($act->obj['type'] === 'Image' && strpos($s['body'],'zrl=') === false) {
2018-09-26 06:50:35 +00:00
2018-10-14 22:47:57 +00:00
$ptr = null;
2018-09-26 06:50:35 +00:00
2018-10-14 22:47:57 +00:00
if(array_key_exists('url',$act->obj)) {
if(is_array($act->obj['url'])) {
if(array_key_exists(0,$act->obj['url'])) {
$ptr = $act->obj['url'];
}
else {
$ptr = [ $act->obj['url'] ];
}
foreach($ptr as $vurl) {
if(strpos($s['body'],$vurl['href']) === false) {
$s['body'] .= "\n\n" . '[zmg]' . $vurl['href'] . '[/zmg]';
break;
}
2018-09-26 06:50:35 +00:00
}
}
2018-10-14 22:47:57 +00:00
elseif(is_string($act->obj['url'])) {
if(strpos($s['body'],$act->obj['url']) === false) {
$s['body'] .= "\n\n" . '[zmg]' . $act->obj['url'] . '[/zmg]';
}
2018-09-20 05:44:22 +00:00
}
}
}
if($act->obj['type'] === 'Page' && ! $s['body']) {
$ptr = null;
$purl = EMPTY_STR;
if(array_key_exists('url',$act->obj)) {
if(is_array($act->obj['url'])) {
if(array_key_exists(0,$act->obj['url'])) {
$ptr = $act->obj['url'];
}
else {
$ptr = [ $act->obj['url'] ];
}
foreach($ptr as $vurl) {
if(array_key_exists('mediaType',$vurl) && $vurl['mediaType'] === 'text/html') {
$purl = $vurl['href'];
break;
}
2018-11-13 03:38:21 +00:00
elseif(array_key_exists('mimeType',$vurl) && $vurl['mimeType'] === 'text/html') {
$purl = $vurl['href'];
2018-11-13 03:38:21 +00:00
break;
}
}
}
elseif(is_string($act->obj['url'])) {
$purl = $act->obj['url'];
}
if($purl) {
$li = z_fetch_url(z_root() . '/linkinfo?binurl=' . bin2hex($purl));
if($li['success'] && $li['body']) {
$s['body'] .= "\n" . $li['body'];
}
else {
$s['body'] .= "\n\n" . $purl;
}
}
}
}
2018-09-20 05:44:22 +00:00
}
2018-10-14 22:47:57 +00:00
if(in_array($act->obj['type'],[ 'Note','Article','Page' ])) {
2018-09-26 06:50:35 +00:00
$ptr = null;
if(array_key_exists('url',$act->obj)) {
if(is_array($act->obj['url'])) {
if(array_key_exists(0,$act->obj['url'])) {
$ptr = $act->obj['url'];
}
else {
$ptr = [ $act->obj['url'] ];
}
foreach($ptr as $vurl) {
2018-10-08 03:37:34 +00:00
if(array_key_exists('mediaType',$vurl) && $vurl['mediaType'] === 'text/html') {
2018-09-26 06:50:35 +00:00
$s['plink'] = $vurl['href'];
break;
}
}
}
elseif(is_string($act->obj['url'])) {
$s['plink'] = $act->obj['url'];
}
}
}
2018-09-20 05:44:22 +00:00
2018-09-26 06:50:35 +00:00
if(! $s['plink']) {
$s['plink'] = $s['mid'];
}
2018-09-20 05:44:22 +00:00
2018-05-30 06:30:18 +00:00
if($act->recips && (! in_array(ACTIVITY_PUBLIC_INBOX,$act->recips)))
$s['item_private'] = 1;
set_iconfig($s,'activitypub','recips',$act->raw_recips);
if($parent) {
set_iconfig($s,'activitypub','rawmsg',$act->raw,1);
}
return $s;
}
2018-09-01 22:52:28 +00:00
static function store($channel,$observer_hash,$act,$item,$fetch_parents = true) {
2018-08-20 03:39:23 +00:00
$is_sys_channel = is_sys_channel($channel['channel_id']);
2018-11-19 05:53:09 +00:00
$is_child_node = false;
2018-08-20 03:39:23 +00:00
// Mastodon only allows visibility in public timelines if the public inbox is listed in the 'to' field.
// They are hidden in the public timeline if the public inbox is listed in the 'cc' field.
// This is not part of the activitypub protocol - we might change this to show all public posts in pubstream at some point.
2018-11-06 02:47:21 +00:00
$pubstream = ((is_array($act->obj) && array_key_exists('to', $act->obj) && is_array($act->obj['to']) && in_array(ACTIVITY_PUBLIC_INBOX, $act->obj['to'])) ? true : false);
2018-08-20 03:39:23 +00:00
2018-11-19 05:53:09 +00:00
if($item['parent_mid'] && $item['parent_mid'] !== $item['mid']) {
$is_child_node = true;
}
$allowed = false;
if ($is_child_node) {
// in ActivityPub, anybody can post comments
$allowed = true;
2018-11-19 05:53:09 +00:00
}
elseif (perm_is_allowed($channel['channel_id'],$observer_hash,'send_stream') || ($is_sys_channel && $pubstream)) {
$allowed = true;
}
2018-11-22 01:46:08 +00:00
if (tgroup_check($channel['channel_id'],$item) && (! $is_child_node)) {
// for forum deliveries, make sure we keep a copy of the signed original
set_iconfig($item,'activitypub','rawmsg',$act->raw,1);
$allowed = true;
}
if(intval($channel['channel_system'])) {
if(! check_pubstream_channelallowed($observer_hash)) {
$allowed = false;
}
// don't allow pubstream posts if the sender even has a clone on a pubstream blacklisted site
$h = q("select hubloc_url from hubloc where hubloc_hash = '%s'",
dbesc($observer_hash)
);
if($h) {
foreach($h as $hub) {
if(! check_pubstream_siteallowed($hub['hubloc_url'])) {
$allowed = false;
break;
}
}
}
}
2018-11-22 01:46:08 +00:00
2018-11-19 05:53:09 +00:00
if(! $allowed) {
2018-08-20 03:39:23 +00:00
logger('no permission');
return;
}
2018-09-01 22:52:28 +00:00
if(is_array($act->obj)) {
$content = self::get_content($act->obj);
}
2018-08-20 03:39:23 +00:00
if(! $content) {
logger('no content');
return;
}
$item['aid'] = $channel['channel_account_id'];
$item['uid'] = $channel['channel_id'];
// Some authors may be zot6 authors in which case we want to store their nomadic identity
// instead of their ActivityPub identity
$item['author_xchan'] = self::find_best_identity($item['author_xchan']);
$item['owner_xchan'] = self::find_best_identity($item['owner_xchan']);
2018-09-04 04:30:39 +00:00
if(! ( $item['author_xchan'] && $item['owner_xchan'])) {
logger('owner or author missing.');
return;
}
2018-08-20 03:39:23 +00:00
if($channel['channel_system']) {
if(! MessageFilter::evaluate($item,get_config('system','pubstream_incl'),get_config('system','pubstream_excl'))) {
2018-08-20 03:39:23 +00:00
logger('post is filtered');
return;
}
}
$abook = q("select * from abook where abook_xchan = '%s' and abook_channel = %d limit 1",
dbesc($observer_hash),
intval($channel['channel_id'])
);
2018-12-11 23:57:15 +00:00
if(! post_is_importable($channel['channel_id'],$item,$abook[0])) {
logger('post is filtered');
return;
}
2018-08-20 03:39:23 +00:00
if($act->obj['conversation']) {
set_iconfig($item,'ostatus','conversation',$act->obj['conversation'],1);
}
2018-09-05 23:30:46 +00:00
// This isn't perfect but the best we can do for now.
$item['comment_policy'] = 'authenticated';
2018-08-20 03:39:23 +00:00
set_iconfig($item,'activitypub','recips',$act->raw_recips);
2018-11-19 05:53:09 +00:00
if(! (isset($act->data['inheritPrivacy']) && $act->data['inheritPrivacy'])) {
if($item['item_private']) {
2018-11-21 02:55:33 +00:00
$item['item_restrict'] = $item['item_restrict'] & 1;
if($is_child_node) {
$item['allow_cid'] = '<' . $channel['channel_hash'] . '>';
$item['allow_gid'] = $item['deny_cid'] = $item['deny_gid'] = '';
}
2018-11-19 05:53:09 +00:00
logger('restricted');
}
}
if(intval($act->sigok)) {
$item['item_verified'] = 1;
}
2018-11-19 05:53:09 +00:00
if($is_child_node) {
$p = q("select parent_mid from item where mid = '%s' and uid = %d limit 1",
dbesc($item['parent_mid']),
intval($item['uid'])
);
if(! $p) {
2018-12-11 07:53:03 +00:00
$a = false;
if(PConfig::Get($channel['channel_id'],'system','hyperdrive',true) || $act->type === 'Announce') {
2018-12-11 07:53:03 +00:00
$a = (($fetch_parents) ? self::fetch_and_store_parents($channel,$observer_hash,$act,$item) : false);
}
2018-09-04 04:30:39 +00:00
if($a) {
$p = q("select parent_mid from item where mid = '%s' and uid = %d limit 1",
dbesc($item['parent_mid']),
intval($item['uid'])
);
}
else {
// if no parent was fetched, turn into a top-level post
2018-08-31 02:19:32 +00:00
// @TODO we maybe could accept these is we formatted the body correctly with share_bb()
// or at least provided a link to the object
2018-10-14 22:47:57 +00:00
if(in_array($act->type,[ 'Like','Dislike','Announce' ])) {
2018-08-31 02:19:32 +00:00
return;
}
// turn into a top level post
$item['parent_mid'] = $item['mid'];
$item['thr_parent'] = $item['mid'];
}
}
if($p[0]['parent_mid'] !== $item['parent_mid']) {
$item['thr_parent'] = $item['parent_mid'];
}
else {
$item['thr_parent'] = $p[0]['parent_mid'];
}
$item['parent_mid'] = $p[0]['parent_mid'];
}
2018-09-04 11:06:50 +00:00
$r = q("select id, created, edited from item where mid = '%s' and uid = %d limit 1",
2018-08-20 03:39:23 +00:00
dbesc($item['mid']),
intval($item['uid'])
);
if($r) {
if($item['edited'] > $r[0]['edited']) {
2018-09-04 11:06:50 +00:00
$item['id'] = $r[0]['id'];
2018-08-20 03:39:23 +00:00
$x = item_store_update($item);
}
else {
return;
}
}
else {
$x = item_store($item);
}
if(is_array($x) && $x['item_id']) {
2018-11-19 05:53:09 +00:00
if($is_child_node) {
2018-08-20 03:39:23 +00:00
if($item['owner_xchan'] === $channel['channel_hash']) {
// We are the owner of this conversation, so send all received comments back downstream
Master::Summon(array('Notifier','comment-import',$x['item_id']));
2018-08-20 03:39:23 +00:00
}
$r = q("select * from item where id = %d limit 1",
intval($x['item_id'])
);
if($r) {
send_status_notifications($x['item_id'],$r[0]);
}
}
sync_an_item($channel['channel_id'],$x['item_id']);
}
}
static public function find_best_identity($xchan) {
$r = q("select hubloc_hash from hubloc where hubloc_id_url = '%s' limit 1",
dbesc($xchan)
);
if($r) {
return $r[0]['hubloc_hash'];
}
return $xchan;
}
2018-09-01 22:52:28 +00:00
static public function fetch_and_store_parents($channel,$observer_hash,$act,$item) {
2018-09-04 01:22:31 +00:00
logger('fetching parents');
2018-09-01 22:52:28 +00:00
$p = [];
$current_act = $act;
$current_item = $item;
2018-08-20 03:39:23 +00:00
2018-09-01 22:52:28 +00:00
while($current_item['parent_mid'] !== $current_item['mid']) {
$n = self::fetch($current_item['parent_mid']);
2018-09-01 22:52:28 +00:00
if(! $n) {
break;
}
$a = new ActivityStreams($n);
2018-09-04 01:22:31 +00:00
logger($a->debug());
2018-09-01 22:52:28 +00:00
if(! $a->is_valid()) {
break;
}
if(is_array($a->actor) && array_key_exists('id',$a->actor)) {
Activity::actor_store($a->actor['id'],$a->actor);
}
$item = null;
switch($a->type) {
case 'Create':
case 'Update':
case 'Like':
case 'Dislike':
case 'Announce':
$item = Activity::decode_note($a);
break;
default:
break;
}
if(! $item) {
break;
}
2018-09-03 03:16:33 +00:00
array_unshift($p,[ $a, $item ]);
2018-09-01 22:52:28 +00:00
if($item['parent_mid'] === $item['mid'] || count($p) > 30) {
2018-09-01 22:52:28 +00:00
break;
}
$current_act = $a;
$current_item = $item;
}
if($p) {
foreach($p as $pv) {
if($pv[0]->is_valid()) {
Activity::store($channel,$observer_hash,$pv[0],$pv[1],false);
}
2018-09-01 22:52:28 +00:00
}
return true;
}
return false;
}
2018-08-20 03:39:23 +00:00
2018-05-30 06:30:18 +00:00
2018-05-30 04:08:52 +00:00
static function announce_note($channel,$observer_hash,$act) {
$s = [];
$is_sys_channel = is_sys_channel($channel['channel_id']);
// Mastodon only allows visibility in public timelines if the public inbox is listed in the 'to' field.
// They are hidden in the public timeline if the public inbox is listed in the 'cc' field.
// This is not part of the activitypub protocol - we might change this to show all public posts in pubstream at some point.
$pubstream = ((is_array($act->obj) && array_key_exists('to', $act->obj) && in_array(ACTIVITY_PUBLIC_INBOX, $act->obj['to'])) ? true : false);
if(! perm_is_allowed($channel['channel_id'],$observer_hash,'send_stream') && ! ($is_sys_channel && $pubstream)) {
logger('no permission');
return;
}
2018-09-01 22:52:28 +00:00
if(is_array($act->obj)) {
$content = self::get_content($act->obj);
}
2018-05-30 04:08:52 +00:00
if(! $content) {
logger('no content');
return;
}
$s['owner_xchan'] = $s['author_xchan'] = $observer_hash;
$s['aid'] = $channel['channel_account_id'];
$s['uid'] = $channel['channel_id'];
$s['mid'] = urldecode($act->obj['id']);
2018-06-04 00:49:48 +00:00
$s['plink'] = urldecode($act->obj['id']);
2018-05-30 04:08:52 +00:00
if(! $s['created'])
$s['created'] = datetime_convert();
if(! $s['edited'])
$s['edited'] = $s['created'];
$s['parent_mid'] = $s['mid'];
$s['verb'] = ACTIVITY_POST;
$s['obj_type'] = ACTIVITY_OBJ_NOTE;
$s['app'] = t('ActivityPub');
if($channel['channel_system']) {
if(! MessageFilter::evaluate($s,get_config('system','pubstream_incl'),get_config('system','pubstream_excl'))) {
2018-05-30 04:08:52 +00:00
logger('post is filtered');
return;
}
}
$abook = q("select * from abook where abook_xchan = '%s' and abook_channel = %d limit 1",
dbesc($observer_hash),
intval($channel['channel_id'])
);
2018-12-11 23:57:15 +00:00
if(! post_is_importable($channel['channel_id'],$s,$abook[0])) {
logger('post is filtered');
return;
2018-05-30 04:08:52 +00:00
}
if($act->obj['conversation']) {
set_iconfig($s,'ostatus','conversation',$act->obj['conversation'],1);
}
$a = self::decode_taxonomy($act->obj);
if($a) {
$s['term'] = $a;
}
$a = self::decode_attachment($act->obj);
if($a) {
$s['attach'] = $a;
}
$body = "[share author='" . urlencode($act->sharee['name']) .
"' profile='" . $act->sharee['url'] .
"' avatar='" . $act->sharee['photo_s'] .
"' link='" . ((is_array($act->obj['url'])) ? $act->obj['url']['href'] : $act->obj['url']) .
"' auth='" . ((is_matrix_url($act->obj['url'])) ? 'true' : 'false' ) .
"' posted='" . $act->obj['published'] .
"' message_id='" . $act->obj['id'] .
"']";
if($content['name'])
$body .= self::bb_content($content,'name') . "\r\n";
$body .= self::bb_content($content,'content');
if($act->obj['type'] === 'Note' && $s['attach']) {
2018-10-11 04:36:55 +00:00
$body .= self::bb_attach($s['attach'],body);
2018-05-30 04:08:52 +00:00
}
$body .= "[/share]";
$s['title'] = self::bb_content($content,'name');
$s['body'] = $body;
if($act->recips && (! in_array(ACTIVITY_PUBLIC_INBOX,$act->recips)))
$s['item_private'] = 1;
set_iconfig($s,'activitypub','recips',$act->raw_recips);
$r = q("select created, edited from item where mid = '%s' and uid = %d limit 1",
dbesc($s['mid']),
intval($s['uid'])
);
if($r) {
if($s['edited'] > $r[0]['edited']) {
$x = item_store_update($s);
}
else {
return;
}
}
else {
$x = item_store($s);
}
if(is_array($x) && $x['item_id']) {
if($parent) {
if($s['owner_xchan'] === $channel['channel_hash']) {
// We are the owner of this conversation, so send all received comments back downstream
Master::Summon(array('Notifier','comment-import',$x['item_id']));
2018-05-30 04:08:52 +00:00
}
$r = q("select * from item where id = %d limit 1",
intval($x['item_id'])
);
if($r) {
send_status_notifications($x['item_id'],$r[0]);
}
}
sync_an_item($channel['channel_id'],$x['item_id']);
}
}
static function like_note($channel,$observer_hash,$act) {
$s = [];
$parent = $act->obj['id'];
if($act->type === 'Like')
$s['verb'] = ACTIVITY_LIKE;
if($act->type === 'Dislike')
$s['verb'] = ACTIVITY_DISLIKE;
if(! $parent)
return;
$r = q("select * from item where uid = %d and ( mid = '%s' or mid = '%s' ) limit 1",
intval($channel['channel_id']),
dbesc($parent),
dbesc(urldecode(basename($parent)))
);
if(! $r) {
logger('parent not found.');
return;
}
xchan_query($r);
$parent_item = $r[0];
if($parent_item['owner_xchan'] === $channel['channel_hash']) {
if(! perm_is_allowed($channel['channel_id'],$observer_hash,'post_comments')) {
logger('no comment permission.');
return;
}
}
if($parent_item['mid'] === $parent_item['parent_mid']) {
$s['parent_mid'] = $parent_item['mid'];
}
else {
$s['thr_parent'] = $parent_item['mid'];
$s['parent_mid'] = $parent_item['parent_mid'];
}
$s['owner_xchan'] = $parent_item['owner_xchan'];
$s['author_xchan'] = $observer_hash;
$s['aid'] = $channel['channel_account_id'];
$s['uid'] = $channel['channel_id'];
$s['mid'] = $act->id;
if(! $s['parent_mid'])
$s['parent_mid'] = $s['mid'];
$post_type = (($parent_item['resource_type'] === 'photo') ? t('photo') : t('status'));
$links = array(array('rel' => 'alternate','type' => 'text/html', 'href' => $parent_item['plink']));
$objtype = (($parent_item['resource_type'] === 'photo') ? ACTIVITY_OBJ_PHOTO : ACTIVITY_OBJ_NOTE );
$body = $parent_item['body'];
$z = q("select * from xchan where xchan_hash = '%s' limit 1",
dbesc($parent_item['author_xchan'])
);
if($z)
$item_author = $z[0];
$object = json_encode(array(
'type' => $post_type,
'id' => $parent_item['mid'],
'parent' => (($parent_item['thr_parent']) ? $parent_item['thr_parent'] : $parent_item['parent_mid']),
'link' => $links,
'title' => $parent_item['title'],
'content' => $parent_item['body'],
'created' => $parent_item['created'],
'edited' => $parent_item['edited'],
'author' => array(
'name' => $item_author['xchan_name'],
'address' => $item_author['xchan_addr'],
'guid' => $item_author['xchan_guid'],
'guid_sig' => $item_author['xchan_guid_sig'],
'link' => array(
array('rel' => 'alternate', 'type' => 'text/html', 'href' => $item_author['xchan_url']),
array('rel' => 'photo', 'type' => $item_author['xchan_photo_mimetype'], 'href' => $item_author['xchan_photo_m'])),
),
), JSON_UNESCAPED_SLASHES
);
if($act->type === 'Like')
$bodyverb = t('%1$s likes %2$s\'s %3$s');
if($act->type === 'Dislike')
$bodyverb = t('%1$s doesn\'t like %2$s\'s %3$s');
$ulink = '[url=' . $item_author['xchan_url'] . ']' . $item_author['xchan_name'] . '[/url]';
$alink = '[url=' . $parent_item['author']['xchan_url'] . ']' . $parent_item['author']['xchan_name'] . '[/url]';
$plink = '[url='. z_root() . '/display/' . urlencode($act->id) . ']' . $post_type . '[/url]';
$s['body'] = sprintf( $bodyverb, $ulink, $alink, $plink );
$s['app'] = t('ActivityPub');
// set the route to that of the parent so downstream hubs won't reject it.
$s['route'] = $parent_item['route'];
$s['item_private'] = $parent_item['item_private'];
$s['obj_type'] = $objtype;
$s['obj'] = $object;
if($act->obj['conversation']) {
set_iconfig($s,'ostatus','conversation',$act->obj['conversation'],1);
}
if($act->recips && (! in_array(ACTIVITY_PUBLIC_INBOX,$act->recips)))
$s['item_private'] = 1;
set_iconfig($s,'activitypub','recips',$act->raw_recips);
$result = item_store($s);
if($result['success']) {
// if the message isn't already being relayed, notify others
if(intval($parent_item['item_origin']))
Master::Summon(array('Notifier','comment-import',$result['item_id']));
sync_an_item($channel['channel_id'],$result['item_id']);
2018-05-30 04:08:52 +00:00
}
return;
}
2018-10-11 04:36:55 +00:00
static function bb_attach($attach,$body) {
2018-05-30 04:08:52 +00:00
$ret = false;
foreach($attach as $a) {
if(strpos($a['type'],'image') !== false) {
2018-10-11 23:54:31 +00:00
if(self::media_not_in_body($a['href'],$body)) {
2018-10-11 04:36:55 +00:00
$ret .= "\n\n" . '[img]' . $a['href'] . '[/img]';
}
2018-05-30 04:08:52 +00:00
}
if(array_key_exists('type',$a) && strpos($a['type'], 'video') === 0) {
2018-10-11 23:54:31 +00:00
if(self::media_not_in_body($a['href'],$body)) {
2018-10-11 04:36:55 +00:00
$ret .= "\n\n" . '[video]' . $a['href'] . '[/video]';
}
2018-05-30 04:08:52 +00:00
}
if(array_key_exists('type',$a) && strpos($a['type'], 'audio') === 0) {
2018-10-11 23:54:31 +00:00
if(self::media_not_in_body($a['href'],$body)) {
2018-10-11 04:36:55 +00:00
$ret .= "\n\n" . '[audio]' . $a['href'] . '[/audio]';
}
2018-05-30 04:08:52 +00:00
}
}
return $ret;
}
2018-10-11 23:54:31 +00:00
// check for the existence of existing media link in body
static function media_not_in_body($s,$body) {
if((strpos($body,']' . $s . '[/img]') === false) &&
(strpos($body,']' . $s . '[/zmg]') === false) &&
2018-10-11 23:54:31 +00:00
(strpos($body,']' . $s . '[/video]') === false) &&
(strpos($body,']' . $s . '[/audio]') === false)) {
return true;
}
return false;
}
2018-05-30 04:08:52 +00:00
static function bb_content($content,$field) {
require_once('include/html2bbcode.php');
2018-08-23 01:45:53 +00:00
require_once('include/event.php');
2018-05-30 04:08:52 +00:00
$ret = false;
if(is_array($content[$field])) {
foreach($content[$field] as $k => $v) {
2018-10-23 22:36:48 +00:00
$ret .= html2bbcode($v);
// save this for auto-translate or dynamic filtering
// $ret .= '[language=' . $k . ']' . html2bbcode($v) . '[/language]';
2018-05-30 04:08:52 +00:00
}
}
else {
2018-07-09 05:28:15 +00:00
if($field === 'bbcode' && array_key_exists('bbcode',$content)) {
$ret = $content[$field];
}
else {
$ret = html2bbcode($content[$field]);
}
2018-05-30 04:08:52 +00:00
}
2018-08-23 01:45:53 +00:00
if($field === 'content' && $content['event'] && (! strpos($ret,'[event'))) {
$ret .= format_event_bbcode($content['event']);
}
2018-05-30 04:08:52 +00:00
return $ret;
}
static function get_content($act) {
$content = [];
2018-08-23 01:45:53 +00:00
$event = null;
2018-09-01 22:52:28 +00:00
if ((! $act) || (! is_array($act))) {
return $content;
}
2018-05-30 04:08:52 +00:00
2018-08-23 01:45:53 +00:00
if($act['type'] === 'Event') {
$adjust = false;
$event = [];
$event['event_hash'] = $act['id'];
if(array_key_exists('startTime',$act) && strpos($act['startTime'],-1,1) === 'Z') {
$adjust = true;
$event['adjust'] = 1;
$event['dtstart'] = datetime_convert('UTC','UTC',$event['startTime'] . (($adjust) ? '' : 'Z'));
}
if(array_key_exists('endTime',$act)) {
$event['dtend'] = datetime_convert('UTC','UTC',$event['endTime'] . (($adjust) ? '' : 'Z'));
}
else {
$event['nofinish'] = true;
}
}
foreach ([ 'name', 'summary', 'content' ] as $a) {
if (($x = self::get_textfield($act,$a)) !== false) {
2018-05-30 04:08:52 +00:00
$content[$a] = $x;
}
}
2018-08-23 01:45:53 +00:00
if($event) {
$event['summary'] = html2bbcode($content['summary']);
$event['description'] = html2bbcode($content['content']);
if($event['summary'] && $event['dtstart']) {
$content['event'] = $event;
}
}
if (array_key_exists('source',$act) && array_key_exists('mediaType',$act['source'])) {
if ($act['source']['mediaType'] === 'text/bbcode') {
2018-07-09 05:28:15 +00:00
$content['bbcode'] = purify_html($act['source']['content']);
}
}
2018-05-30 04:08:52 +00:00
2018-08-23 01:45:53 +00:00
2018-05-30 04:08:52 +00:00
return $content;
}
static function get_textfield($act,$field) {
$content = false;
if(array_key_exists($field,$act) && $act[$field])
$content = purify_html($act[$field]);
elseif(array_key_exists($field . 'Map',$act) && $act[$field . 'Map']) {
foreach($act[$field . 'Map'] as $k => $v) {
$content[escape_tags($k)] = purify_html($v);
}
}
return $content;
}
2018-08-23 06:04:37 +00:00
}